2006-07-11 22:01:26 +04:00
/*
Unix SMB / CIFS implementation .
routines for marshalling / unmarshalling string types
Copyright ( C ) Andrew Tridgell 2003
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
2007-07-09 23:25:36 +04:00
the Free Software Foundation ; either version 3 of the License , or
2006-07-11 22:01:26 +04:00
( 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
2007-07-10 04:52:41 +04:00
along with this program . If not , see < http : //www.gnu.org/licenses/>.
2006-07-11 22:01:26 +04:00
*/
# include "includes.h"
2007-11-08 16:36:13 +03:00
# include "librpc/ndr/libndr.h"
2006-07-11 22:01:26 +04:00
/**
pull a general string from the wire
*/
2007-11-09 22:23:40 +03:00
_PUBLIC_ enum ndr_err_code ndr_pull_string ( struct ndr_pull * ndr , int ndr_flags , const char * * s )
2006-07-11 22:01:26 +04:00
{
char * as = NULL ;
uint32_t len1 , ofs , len2 ;
uint16_t len3 ;
2011-05-31 11:49:18 +04:00
size_t conv_src_len = 0 , converted_size ;
2011-05-31 11:49:19 +04:00
int do_convert = 1 , chset = CH_UTF16 ;
2006-07-11 22:01:26 +04:00
unsigned byte_mul = 2 ;
unsigned flags = ndr - > flags ;
unsigned c_len_term = 0 ;
if ( ! ( ndr_flags & NDR_SCALARS ) ) {
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2006-09-21 22:37:09 +04:00
if ( NDR_BE ( ndr ) ) {
chset = CH_UTF16BE ;
}
2006-07-11 22:01:26 +04:00
if ( flags & LIBNDR_FLAG_STR_ASCII ) {
chset = CH_DOS ;
byte_mul = 1 ;
flags & = ~ LIBNDR_FLAG_STR_ASCII ;
}
if ( flags & LIBNDR_FLAG_STR_UTF8 ) {
chset = CH_UTF8 ;
byte_mul = 1 ;
flags & = ~ LIBNDR_FLAG_STR_UTF8 ;
}
2011-05-31 11:49:19 +04:00
if ( flags & LIBNDR_FLAG_STR_RAW8 ) {
do_convert = 0 ;
byte_mul = 1 ;
flags & = ~ LIBNDR_FLAG_STR_RAW8 ;
}
2006-07-11 22:01:26 +04:00
flags & = ~ LIBNDR_FLAG_STR_CONFORMANT ;
if ( flags & LIBNDR_FLAG_STR_CHARLEN ) {
c_len_term = 1 ;
flags & = ~ LIBNDR_FLAG_STR_CHARLEN ;
}
switch ( flags & LIBNDR_STRING_FLAGS ) {
case LIBNDR_FLAG_STR_LEN4 | LIBNDR_FLAG_STR_SIZE4 :
case LIBNDR_FLAG_STR_LEN4 | LIBNDR_FLAG_STR_SIZE4 | LIBNDR_FLAG_STR_NOTERM :
NDR_CHECK ( ndr_pull_uint32 ( ndr , NDR_SCALARS , & len1 ) ) ;
NDR_CHECK ( ndr_pull_uint32 ( ndr , NDR_SCALARS , & ofs ) ) ;
if ( ofs ! = 0 ) {
return ndr_pull_error ( ndr , NDR_ERR_STRING , " non-zero array offset with string flags 0x%x \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
}
NDR_CHECK ( ndr_pull_uint32 ( ndr , NDR_SCALARS , & len2 ) ) ;
if ( len2 > len1 ) {
2011-05-31 11:49:18 +04:00
return ndr_pull_error ( ndr , NDR_ERR_STRING ,
" Bad string lengths len1=%u ofs=%u len2=%u \n " ,
2006-07-11 22:01:26 +04:00
len1 , ofs , len2 ) ;
2011-05-31 11:49:18 +04:00
} else if ( len1 ! = len2 ) {
2006-07-11 22:01:26 +04:00
DEBUG ( 6 , ( " len1[%u] != len2[%u] '%s' \n " , len1 , len2 , as ) ) ;
}
2011-05-31 11:49:18 +04:00
conv_src_len = len2 + c_len_term ;
2006-07-11 22:01:26 +04:00
break ;
case LIBNDR_FLAG_STR_SIZE4 :
case LIBNDR_FLAG_STR_SIZE4 | LIBNDR_FLAG_STR_NOTERM :
NDR_CHECK ( ndr_pull_uint32 ( ndr , NDR_SCALARS , & len1 ) ) ;
2011-05-31 11:49:18 +04:00
conv_src_len = len1 + c_len_term ;
2006-07-11 22:01:26 +04:00
break ;
case LIBNDR_FLAG_STR_LEN4 :
case LIBNDR_FLAG_STR_LEN4 | LIBNDR_FLAG_STR_NOTERM :
NDR_CHECK ( ndr_pull_uint32 ( ndr , NDR_SCALARS , & ofs ) ) ;
if ( ofs ! = 0 ) {
return ndr_pull_error ( ndr , NDR_ERR_STRING , " non-zero array offset with string flags 0x%x \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
}
NDR_CHECK ( ndr_pull_uint32 ( ndr , NDR_SCALARS , & len1 ) ) ;
2011-05-31 11:49:18 +04:00
conv_src_len = len1 + c_len_term ;
2006-07-11 22:01:26 +04:00
break ;
case LIBNDR_FLAG_STR_SIZE2 :
case LIBNDR_FLAG_STR_SIZE2 | LIBNDR_FLAG_STR_NOTERM :
NDR_CHECK ( ndr_pull_uint16 ( ndr , NDR_SCALARS , & len3 ) ) ;
2011-05-31 11:49:18 +04:00
conv_src_len = len3 + c_len_term ;
2006-07-11 22:01:26 +04:00
break ;
case LIBNDR_FLAG_STR_SIZE2 | LIBNDR_FLAG_STR_NOTERM | LIBNDR_FLAG_STR_BYTESIZE :
NDR_CHECK ( ndr_pull_uint16 ( ndr , NDR_SCALARS , & len3 ) ) ;
2011-05-31 11:49:18 +04:00
conv_src_len = len3 ;
byte_mul = 1 ; /* the length is now absolute */
2006-07-11 22:01:26 +04:00
break ;
case LIBNDR_FLAG_STR_NULLTERM :
2019-11-07 04:19:24 +03:00
/*
* We ensure that conv_str_len cannot return 0 by
* requring that there be enough bytes for at least
* the NULL terminator
*/
2006-07-11 22:01:26 +04:00
if ( byte_mul = = 1 ) {
2019-11-07 04:19:24 +03:00
NDR_PULL_NEED_BYTES ( ndr , 1 ) ;
2011-05-31 11:49:18 +04:00
conv_src_len = ascii_len_n ( ( const char * ) ( ndr - > data + ndr - > offset ) , ndr - > data_size - ndr - > offset ) ;
2006-07-11 22:01:26 +04:00
} else {
2019-11-07 04:19:24 +03:00
NDR_PULL_NEED_BYTES ( ndr , 2 ) ;
2011-05-31 11:49:18 +04:00
conv_src_len = utf16_len_n ( ndr - > data + ndr - > offset , ndr - > data_size - ndr - > offset ) ;
2006-07-11 22:01:26 +04:00
}
2011-05-31 11:49:18 +04:00
byte_mul = 1 ; /* the length is now absolute */
2006-07-11 22:01:26 +04:00
break ;
2007-11-08 16:36:13 +03:00
case LIBNDR_FLAG_STR_NOTERM :
if ( ! ( ndr - > flags & LIBNDR_FLAG_REMAINING ) ) {
return ndr_pull_error ( ndr , NDR_ERR_STRING , " Bad string flags 0x%x (missing NDR_REMAINING) \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
}
2011-05-31 11:49:18 +04:00
conv_src_len = ndr - > data_size - ndr - > offset ;
byte_mul = 1 ; /* the length is now absolute */
2007-11-08 16:36:13 +03:00
break ;
2006-07-11 22:01:26 +04:00
default :
return ndr_pull_error ( ndr , NDR_ERR_STRING , " Bad string flags 0x%x \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
}
2011-05-31 11:49:18 +04:00
NDR_PULL_NEED_BYTES ( ndr , conv_src_len * byte_mul ) ;
if ( conv_src_len = = 0 ) {
as = talloc_strdup ( ndr - > current_mem_ctx , " " ) ;
2011-05-31 14:21:37 +04:00
converted_size = 0 ;
2011-05-31 11:49:18 +04:00
} else {
2011-05-31 11:49:19 +04:00
if ( ! do_convert ) {
as = talloc_strndup ( ndr - > current_mem_ctx ,
2011-06-02 01:49:03 +04:00
( char * ) ndr - > data + ndr - > offset ,
2011-05-31 11:49:19 +04:00
conv_src_len ) ;
2011-05-31 14:21:37 +04:00
if ( ! as ) {
return ndr_pull_error ( ndr , NDR_ERR_ALLOC ,
" Failed to talloc_strndup() in RAW8 ndr_string_pull() " ) ;
}
converted_size = MIN ( strlen ( as ) + 1 , conv_src_len ) ;
2011-05-31 11:49:19 +04:00
} else if ( ! convert_string_talloc ( ndr - > current_mem_ctx , chset ,
2011-05-31 11:49:18 +04:00
CH_UNIX , ndr - > data + ndr - > offset ,
conv_src_len * byte_mul ,
( void * * ) ( void * ) & as ,
& converted_size ) ) {
return ndr_pull_error ( ndr , NDR_ERR_CHARCNV ,
" Bad character conversion with flags 0x%x " , flags ) ;
}
}
/* this is a way of detecting if a string is sent with the wrong
termination */
if ( ndr - > flags & LIBNDR_FLAG_STR_NOTERM ) {
2011-05-31 14:21:37 +04:00
if ( as & & converted_size > 0 & & as [ converted_size - 1 ] = = ' \0 ' ) {
DEBUG ( 6 , ( " short string '%s', sent with NULL termination despite NOTERM flag in IDL \n " , as ) ) ;
2011-05-31 11:49:18 +04:00
}
} else {
2011-05-31 14:21:37 +04:00
if ( as & & converted_size > 0 & & as [ converted_size - 1 ] ! = ' \0 ' ) {
DEBUG ( 6 , ( " long string '%s', send without NULL termination (which was expected) \n " , as ) ) ;
2011-05-31 11:49:18 +04:00
}
}
NDR_CHECK ( ndr_pull_advance ( ndr , conv_src_len * byte_mul ) ) ;
* s = as ;
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
/**
push a general string onto the wire
*/
2007-11-09 22:23:40 +03:00
_PUBLIC_ enum ndr_err_code ndr_push_string ( struct ndr_push * ndr , int ndr_flags , const char * s )
2006-07-11 22:01:26 +04:00
{
2008-05-21 01:06:38 +04:00
ssize_t s_len , c_len ;
size_t d_len ;
2011-05-31 11:49:19 +04:00
int do_convert = 1 , chset = CH_UTF16 ;
2006-07-11 22:01:26 +04:00
unsigned flags = ndr - > flags ;
unsigned byte_mul = 2 ;
uint8_t * dest = NULL ;
if ( ! ( ndr_flags & NDR_SCALARS ) ) {
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2006-09-21 22:37:09 +04:00
if ( NDR_BE ( ndr ) ) {
chset = CH_UTF16BE ;
}
2006-07-11 22:01:26 +04:00
s_len = s ? strlen ( s ) : 0 ;
if ( flags & LIBNDR_FLAG_STR_ASCII ) {
chset = CH_DOS ;
byte_mul = 1 ;
flags & = ~ LIBNDR_FLAG_STR_ASCII ;
}
if ( flags & LIBNDR_FLAG_STR_UTF8 ) {
chset = CH_UTF8 ;
byte_mul = 1 ;
flags & = ~ LIBNDR_FLAG_STR_UTF8 ;
}
2011-05-31 11:49:19 +04:00
if ( flags & LIBNDR_FLAG_STR_RAW8 ) {
do_convert = 0 ;
byte_mul = 1 ;
flags & = ~ LIBNDR_FLAG_STR_RAW8 ;
}
2006-07-11 22:01:26 +04:00
flags & = ~ LIBNDR_FLAG_STR_CONFORMANT ;
2008-10-13 17:01:28 +04:00
if ( ! ( flags & LIBNDR_FLAG_STR_NOTERM ) ) {
2006-07-11 22:01:26 +04:00
s_len + + ;
}
2011-05-31 11:49:19 +04:00
2021-11-03 15:57:50 +03:00
if ( s_len = = 0 ) {
d_len = 0 ;
dest = ( uint8_t * ) talloc_strdup ( ndr , " " ) ;
} else if ( ! do_convert ) {
2011-05-31 11:49:19 +04:00
d_len = s_len ;
2011-06-02 01:49:03 +04:00
dest = ( uint8_t * ) talloc_strndup ( ndr , s , s_len ) ;
2011-05-31 11:49:19 +04:00
} else if ( ! convert_string_talloc ( ndr , CH_UNIX , chset , s , s_len ,
2011-03-24 02:59:41 +03:00
( void * * ) ( void * ) & dest , & d_len ) )
2008-04-30 01:36:24 +04:00
{
2006-07-11 22:01:26 +04:00
return ndr_push_error ( ndr , NDR_ERR_CHARCNV ,
2010-05-20 15:54:44 +04:00
" Bad character push conversion with flags 0x%x " , flags ) ;
2006-07-11 22:01:26 +04:00
}
if ( flags & LIBNDR_FLAG_STR_BYTESIZE ) {
c_len = d_len ;
flags & = ~ LIBNDR_FLAG_STR_BYTESIZE ;
} else if ( flags & LIBNDR_FLAG_STR_CHARLEN ) {
c_len = ( d_len / byte_mul ) - 1 ;
flags & = ~ LIBNDR_FLAG_STR_CHARLEN ;
} else {
c_len = d_len / byte_mul ;
}
switch ( ( flags & LIBNDR_STRING_FLAGS ) & ~ LIBNDR_FLAG_STR_NOTERM ) {
case LIBNDR_FLAG_STR_LEN4 | LIBNDR_FLAG_STR_SIZE4 :
NDR_CHECK ( ndr_push_uint32 ( ndr , NDR_SCALARS , c_len ) ) ;
NDR_CHECK ( ndr_push_uint32 ( ndr , NDR_SCALARS , 0 ) ) ;
NDR_CHECK ( ndr_push_uint32 ( ndr , NDR_SCALARS , c_len ) ) ;
NDR_CHECK ( ndr_push_bytes ( ndr , dest , d_len ) ) ;
break ;
case LIBNDR_FLAG_STR_LEN4 :
NDR_CHECK ( ndr_push_uint32 ( ndr , NDR_SCALARS , 0 ) ) ;
NDR_CHECK ( ndr_push_uint32 ( ndr , NDR_SCALARS , c_len ) ) ;
NDR_CHECK ( ndr_push_bytes ( ndr , dest , d_len ) ) ;
break ;
case LIBNDR_FLAG_STR_SIZE4 :
NDR_CHECK ( ndr_push_uint32 ( ndr , NDR_SCALARS , c_len ) ) ;
NDR_CHECK ( ndr_push_bytes ( ndr , dest , d_len ) ) ;
break ;
case LIBNDR_FLAG_STR_SIZE2 :
NDR_CHECK ( ndr_push_uint16 ( ndr , NDR_SCALARS , c_len ) ) ;
NDR_CHECK ( ndr_push_bytes ( ndr , dest , d_len ) ) ;
break ;
case LIBNDR_FLAG_STR_NULLTERM :
NDR_CHECK ( ndr_push_bytes ( ndr , dest , d_len ) ) ;
break ;
default :
2007-11-08 16:36:13 +03:00
if ( ndr - > flags & LIBNDR_FLAG_REMAINING ) {
NDR_CHECK ( ndr_push_bytes ( ndr , dest , d_len ) ) ;
break ;
}
2006-07-11 22:01:26 +04:00
return ndr_push_error ( ndr , NDR_ERR_STRING , " Bad string flags 0x%x \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
}
talloc_free ( dest ) ;
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
/**
push a general string onto the wire
*/
2007-11-08 16:36:13 +03:00
_PUBLIC_ size_t ndr_string_array_size ( struct ndr_push * ndr , const char * s )
2006-07-11 22:01:26 +04:00
{
size_t c_len ;
unsigned flags = ndr - > flags ;
unsigned byte_mul = 2 ;
unsigned c_len_term = 1 ;
2011-05-31 11:49:19 +04:00
if ( flags & LIBNDR_FLAG_STR_RAW8 ) {
c_len = s ? strlen ( s ) : 0 ;
} else {
c_len = s ? strlen_m ( s ) : 0 ;
}
2006-07-11 22:01:26 +04:00
2011-05-31 11:49:19 +04:00
if ( flags & ( LIBNDR_FLAG_STR_ASCII | LIBNDR_FLAG_STR_RAW8 | LIBNDR_FLAG_STR_UTF8 ) ) {
2006-07-11 22:01:26 +04:00
byte_mul = 1 ;
}
if ( flags & LIBNDR_FLAG_STR_NOTERM ) {
c_len_term = 0 ;
}
c_len = c_len + c_len_term ;
if ( flags & LIBNDR_FLAG_STR_BYTESIZE ) {
c_len = c_len * byte_mul ;
}
return c_len ;
}
2007-11-08 16:36:13 +03:00
_PUBLIC_ void ndr_print_string ( struct ndr_print * ndr , const char * name , const char * s )
2006-07-11 22:01:26 +04:00
{
2021-02-10 13:23:13 +03:00
if ( NDR_HIDE_SECRET ( ndr ) ) {
ndr - > print ( ndr , " %-25s: <REDACTED SECRET VALUE> " , name ) ;
return ;
}
2006-07-11 22:01:26 +04:00
if ( s ) {
ndr - > print ( ndr , " %-25s: '%s' " , name , s ) ;
} else {
ndr - > print ( ndr , " %-25s: NULL " , name ) ;
}
}
2007-11-08 16:36:13 +03:00
_PUBLIC_ uint32_t ndr_size_string ( int ret , const char * const * string , int flags )
2006-07-11 22:01:26 +04:00
{
/* FIXME: Is this correct for all strings ? */
if ( ! ( * string ) ) return ret ;
return ret + strlen ( * string ) + 1 ;
}
2020-07-30 03:06:10 +03:00
static uint32_t guess_string_array_size ( struct ndr_pull * ndr , int ndr_flags )
{
/*
* Here we could do something clever like count the number of zeros in
* the ndr data , but it is probably sufficient to pick a lowish number
* ( compared to the overhead of the talloc header ) and let the
* expontential resizing deal with longer arrays .
*/
return 5 ;
}
static enum ndr_err_code extend_string_array ( struct ndr_pull * ndr ,
const char * * * _a ,
uint32_t * count )
{
const char * * a = * _a ;
uint32_t inc = * count / 4 + 3 ;
uint32_t alloc_size = * count + inc ;
if ( alloc_size < * count ) {
/* overflow ! */
return NDR_ERR_ALLOC ;
}
/*
* We allocate and zero two more bytes than we report back , so that
* the string array will always be NULL terminated .
*/
a = talloc_realloc ( ndr - > current_mem_ctx , a ,
const char * ,
alloc_size ) ;
NDR_ERR_HAVE_NO_MEMORY ( a ) ;
memset ( a + * count , 0 , inc * sizeof ( a [ 0 ] ) ) ;
* _a = a ;
* count = alloc_size - 2 ;
return NDR_ERR_SUCCESS ;
}
2006-07-11 22:01:26 +04:00
/**
pull a general string array from the wire
*/
2007-11-09 22:23:40 +03:00
_PUBLIC_ enum ndr_err_code ndr_pull_string_array ( struct ndr_pull * ndr , int ndr_flags , const char * * * _a )
2006-07-11 22:01:26 +04:00
{
2009-02-19 13:11:28 +03:00
const char * * a = NULL ;
2006-07-11 22:01:26 +04:00
uint32_t count ;
2007-11-08 16:36:13 +03:00
unsigned flags = ndr - > flags ;
unsigned saved_flags = ndr - > flags ;
2020-07-30 03:06:10 +03:00
uint32_t alloc_size ;
2006-07-11 22:01:26 +04:00
if ( ! ( ndr_flags & NDR_SCALARS ) ) {
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2020-07-30 03:06:10 +03:00
alloc_size = guess_string_array_size ( ndr , ndr_flags ) ;
a = talloc_zero_array ( ndr - > current_mem_ctx , const char * , alloc_size + 2 ) ;
NDR_ERR_HAVE_NO_MEMORY ( a ) ;
2010-05-13 04:27:29 +04:00
switch ( flags & ( LIBNDR_FLAG_STR_NULLTERM | LIBNDR_FLAG_STR_NOTERM ) ) {
2007-11-08 16:36:13 +03:00
case LIBNDR_FLAG_STR_NULLTERM :
2020-07-30 03:06:10 +03:00
/*
2007-11-08 16:36:13 +03:00
* here the strings are null terminated
2010-09-04 20:08:05 +04:00
* but also the array is null terminated if LIBNDR_FLAG_REMAINING
* is specified
2007-11-08 16:36:13 +03:00
*/
for ( count = 0 ; ; count + + ) {
TALLOC_CTX * tmp_ctx ;
const char * s = NULL ;
2020-07-30 03:06:10 +03:00
if ( count = = alloc_size ) {
NDR_CHECK ( extend_string_array ( ndr ,
& a ,
& alloc_size ) ) ;
}
2007-11-08 16:36:13 +03:00
tmp_ctx = ndr - > current_mem_ctx ;
ndr - > current_mem_ctx = a ;
NDR_CHECK ( ndr_pull_string ( ndr , ndr_flags , & s ) ) ;
2020-07-30 01:46:17 +03:00
ndr - > current_mem_ctx = tmp_ctx ;
2010-09-04 20:08:05 +04:00
if ( ( ndr - > data_size - ndr - > offset ) = = 0 & & ndr - > flags & LIBNDR_FLAG_REMAINING )
{
a [ count ] = s ;
break ;
}
2007-11-08 16:36:13 +03:00
if ( strcmp ( " " , s ) = = 0 ) {
a [ count ] = NULL ;
break ;
} else {
a [ count ] = s ;
}
}
* _a = a ;
break ;
case LIBNDR_FLAG_STR_NOTERM :
if ( ! ( ndr - > flags & LIBNDR_FLAG_REMAINING ) ) {
return ndr_pull_error ( ndr , NDR_ERR_STRING , " Bad string flags 0x%x (missing NDR_REMAINING) \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
}
/*
* here the strings are not null terminated
2020-07-30 03:06:10 +03:00
* but separated by a null terminator
2007-11-08 16:36:13 +03:00
*
* which means the same as :
2010-09-04 20:08:05 +04:00
* Every string is null terminated exept the last
2007-11-08 16:36:13 +03:00
* string is terminated by the end of the buffer
*
* as LIBNDR_FLAG_STR_NULLTERM also end at the end
* of the buffer , we can pull each string with this flag
2010-09-04 20:08:05 +04:00
*
* The big difference with the case LIBNDR_FLAG_STR_NOTERM +
* LIBNDR_FLAG_REMAINING is that the last string will not be null terminated
2007-11-08 16:36:13 +03:00
*/
ndr - > flags & = ~ ( LIBNDR_FLAG_STR_NOTERM | LIBNDR_FLAG_REMAINING ) ;
ndr - > flags | = LIBNDR_FLAG_STR_NULLTERM ;
for ( count = 0 ; ( ( ndr - > data_size - ndr - > offset ) > 0 ) ; count + + ) {
TALLOC_CTX * tmp_ctx ;
const char * s = NULL ;
2020-07-30 03:06:10 +03:00
if ( count = = alloc_size ) {
NDR_CHECK ( extend_string_array ( ndr ,
& a ,
& alloc_size ) ) ;
}
2007-11-08 16:36:13 +03:00
tmp_ctx = ndr - > current_mem_ctx ;
ndr - > current_mem_ctx = a ;
NDR_CHECK ( ndr_pull_string ( ndr , ndr_flags , & s ) ) ;
ndr - > current_mem_ctx = tmp_ctx ;
2006-07-11 22:01:26 +04:00
a [ count ] = s ;
}
2007-11-08 16:36:13 +03:00
2020-07-30 03:06:10 +03:00
a = talloc_realloc ( ndr - > current_mem_ctx , a , const char * , count + 1 ) ;
NDR_ERR_HAVE_NO_MEMORY ( a ) ;
* _a = a ;
2007-11-08 16:36:13 +03:00
break ;
default :
return ndr_pull_error ( ndr , NDR_ERR_STRING , " Bad string flags 0x%x \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
2006-07-11 22:01:26 +04:00
}
2007-11-08 16:36:13 +03:00
ndr - > flags = saved_flags ;
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
/**
push a general string array onto the wire
*/
2007-11-09 22:23:40 +03:00
_PUBLIC_ enum ndr_err_code ndr_push_string_array ( struct ndr_push * ndr , int ndr_flags , const char * * a )
2006-07-11 22:01:26 +04:00
{
uint32_t count ;
2007-11-08 16:36:13 +03:00
unsigned flags = ndr - > flags ;
unsigned saved_flags = ndr - > flags ;
2006-07-11 22:01:26 +04:00
if ( ! ( ndr_flags & NDR_SCALARS ) ) {
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2007-11-08 16:36:13 +03:00
switch ( flags & LIBNDR_STRING_FLAGS ) {
case LIBNDR_FLAG_STR_NULLTERM :
for ( count = 0 ; a & & a [ count ] ; count + + ) {
NDR_CHECK ( ndr_push_string ( ndr , ndr_flags , a [ count ] ) ) ;
}
2010-09-04 20:08:05 +04:00
/* If LIBNDR_FLAG_REMAINING then we do not add a null terminator to the array */
if ( ! ( flags & LIBNDR_FLAG_REMAINING ) )
{
NDR_CHECK ( ndr_push_string ( ndr , ndr_flags , " " ) ) ;
}
2007-11-08 16:36:13 +03:00
break ;
case LIBNDR_FLAG_STR_NOTERM :
if ( ! ( ndr - > flags & LIBNDR_FLAG_REMAINING ) ) {
return ndr_push_error ( ndr , NDR_ERR_STRING , " Bad string flags 0x%x (missing NDR_REMAINING) \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
}
for ( count = 0 ; a & & a [ count ] ; count + + ) {
if ( count > 0 ) {
ndr - > flags & = ~ ( LIBNDR_FLAG_STR_NOTERM | LIBNDR_FLAG_REMAINING ) ;
ndr - > flags | = LIBNDR_FLAG_STR_NULLTERM ;
NDR_CHECK ( ndr_push_string ( ndr , ndr_flags , " " ) ) ;
ndr - > flags = saved_flags ;
}
NDR_CHECK ( ndr_push_string ( ndr , ndr_flags , a [ count ] ) ) ;
}
2006-07-11 22:01:26 +04:00
2007-11-08 16:36:13 +03:00
break ;
default :
return ndr_push_error ( ndr , NDR_ERR_STRING , " Bad string flags 0x%x \n " ,
ndr - > flags & LIBNDR_STRING_FLAGS ) ;
}
ndr - > flags = saved_flags ;
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2007-11-08 16:36:13 +03:00
_PUBLIC_ void ndr_print_string_array ( struct ndr_print * ndr , const char * name , const char * * a )
2006-07-11 22:01:26 +04:00
{
uint32_t count ;
uint32_t i ;
for ( count = 0 ; a & & a [ count ] ; count + + ) { }
ndr - > print ( ndr , " %s: ARRAY(%d) " , name , count ) ;
ndr - > depth + + ;
for ( i = 0 ; i < count ; i + + ) {
char * idx = NULL ;
2008-02-25 17:24:49 +03:00
if ( asprintf ( & idx , " [%d] " , i ) ! = - 1 ) {
2006-07-11 22:01:26 +04:00
ndr_print_string ( ndr , idx , a [ i ] ) ;
free ( idx ) ;
}
}
ndr - > depth - - ;
}
2009-02-02 18:41:44 +03:00
_PUBLIC_ size_t ndr_size_string_array ( const char * * a , uint32_t count , int flags )
{
uint32_t i ;
size_t size = 0 ;
2011-05-31 11:49:19 +04:00
int rawbytes = 0 ;
if ( flags & LIBNDR_FLAG_STR_RAW8 ) {
rawbytes = 1 ;
flags & = ~ LIBNDR_FLAG_STR_RAW8 ;
}
2009-02-02 18:41:44 +03:00
switch ( flags & LIBNDR_STRING_FLAGS ) {
case LIBNDR_FLAG_STR_NULLTERM :
for ( i = 0 ; i < count ; i + + ) {
2011-05-31 11:49:19 +04:00
size + = rawbytes ? strlen ( a [ i ] ) + 1 : strlen_m_term ( a [ i ] ) ;
2009-02-02 18:41:44 +03:00
}
break ;
case LIBNDR_FLAG_STR_NOTERM :
for ( i = 0 ; i < count ; i + + ) {
2011-05-31 11:49:19 +04:00
size + = rawbytes ? strlen ( a [ i ] ) : strlen_m ( a [ i ] ) ;
2009-02-02 18:41:44 +03:00
}
break ;
default :
return 0 ;
}
return size ;
}
2006-07-11 22:01:26 +04:00
/**
* Return number of elements in a string including the last ( zeroed ) element
*/
2007-11-08 16:36:13 +03:00
_PUBLIC_ uint32_t ndr_string_length ( const void * _var , uint32_t element_size )
2006-07-11 22:01:26 +04:00
{
uint32_t i ;
uint8_t zero [ 4 ] = { 0 , 0 , 0 , 0 } ;
const char * var = ( const char * ) _var ;
for ( i = 0 ; memcmp ( var + i * element_size , zero , element_size ) ! = 0 ; i + + ) ;
return i + 1 ;
}
2019-12-16 17:50:17 +03:00
/**
* @ brief Get the string length including the null terminator if available .
*
* This checks the string length based on the elements . The returned number
* includes the terminating null byte ( s ) if found .
*
* @ param [ in ] _var The string the calculate the length for .
*
* @ param [ in ] length The length of the buffer passed by _var .
*
* @ param [ in ] element_size The element_size of a string char in bytes .
*
* @ return The length of the strings or 0.
*/
static uint32_t ndr_string_n_length ( const void * _var ,
size_t length ,
uint32_t element_size )
{
size_t i = 0 ;
uint8_t zero [ 4 ] = { 0 , 0 , 0 , 0 } ;
const char * var = ( const char * ) _var ;
int cmp ;
if ( element_size > 4 ) {
return 0 ;
}
for ( i = 0 ; i < length ; i + + , var + = element_size ) {
cmp = memcmp ( var , zero , element_size ) ;
if ( cmp = = 0 ) {
break ;
}
}
if ( i = = length ) {
return length ;
}
return i + 1 ;
}
2007-11-09 22:23:40 +03:00
_PUBLIC_ enum ndr_err_code ndr_check_string_terminator ( struct ndr_pull * ndr , uint32_t count , uint32_t element_size )
2006-07-11 22:01:26 +04:00
{
uint32_t i ;
2008-10-13 17:01:28 +04:00
uint32_t save_offset ;
2006-07-11 22:01:26 +04:00
2008-10-13 17:01:28 +04:00
save_offset = ndr - > offset ;
2012-12-14 20:09:34 +04:00
NDR_CHECK ( ndr_pull_advance ( ndr , ( count - 1 ) * element_size ) ) ;
2006-07-11 22:01:26 +04:00
NDR_PULL_NEED_BYTES ( ndr , element_size ) ;
for ( i = 0 ; i < element_size ; i + + ) {
if ( ndr - > data [ ndr - > offset + i ] ! = 0 ) {
2008-10-13 17:01:28 +04:00
ndr - > offset = save_offset ;
2006-07-11 22:01:26 +04:00
return ndr_pull_error ( ndr , NDR_ERR_ARRAY_SIZE , " String terminator not present or outside string boundaries " ) ;
}
}
2008-10-13 17:01:28 +04:00
ndr - > offset = save_offset ;
2006-07-11 22:01:26 +04:00
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2007-11-09 22:23:40 +03:00
_PUBLIC_ enum ndr_err_code ndr_pull_charset ( struct ndr_pull * ndr , int ndr_flags , const char * * var , uint32_t length , uint8_t byte_mul , charset_t chset )
2006-07-11 22:01:26 +04:00
{
2008-04-30 01:36:24 +04:00
size_t converted_size ;
2006-07-11 22:01:26 +04:00
if ( length = = 0 ) {
* var = talloc_strdup ( ndr - > current_mem_ctx , " " ) ;
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2006-09-21 22:37:09 +04:00
if ( NDR_BE ( ndr ) & & chset = = CH_UTF16 ) {
chset = CH_UTF16BE ;
}
2006-07-11 22:01:26 +04:00
2018-03-26 13:02:01 +03:00
if ( ( byte_mul ! = 0 ) & & ( length > UINT32_MAX / byte_mul ) ) {
return ndr_pull_error ( ndr , NDR_ERR_BUFSIZE , " length overflow " ) ;
}
2006-07-11 22:01:26 +04:00
NDR_PULL_NEED_BYTES ( ndr , length * byte_mul ) ;
2008-04-30 01:36:24 +04:00
if ( ! convert_string_talloc ( ndr - > current_mem_ctx , chset , CH_UNIX ,
ndr - > data + ndr - > offset , length * byte_mul ,
discard_const_p ( void * , var ) ,
2011-03-24 02:59:41 +03:00
& converted_size ) )
2008-04-30 01:36:24 +04:00
{
2006-07-11 22:01:26 +04:00
return ndr_pull_error ( ndr , NDR_ERR_CHARCNV ,
2008-10-13 17:33:40 +04:00
" Bad character conversion " ) ;
2006-07-11 22:01:26 +04:00
}
NDR_CHECK ( ndr_pull_advance ( ndr , length * byte_mul ) ) ;
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2011-11-13 23:01:43 +04:00
_PUBLIC_ enum ndr_err_code ndr_pull_charset_to_null ( struct ndr_pull * ndr , int ndr_flags , const char * * var , uint32_t length , uint8_t byte_mul , charset_t chset )
{
size_t converted_size ;
uint32_t str_len ;
if ( length = = 0 ) {
* var = talloc_strdup ( ndr - > current_mem_ctx , " " ) ;
return NDR_ERR_SUCCESS ;
}
if ( NDR_BE ( ndr ) & & chset = = CH_UTF16 ) {
chset = CH_UTF16BE ;
}
NDR_PULL_NEED_BYTES ( ndr , length * byte_mul ) ;
2019-12-16 17:50:17 +03:00
str_len = ndr_string_n_length ( ndr - > data + ndr - > offset , length , byte_mul ) ;
if ( str_len = = 0 ) {
return ndr_pull_error ( ndr , NDR_ERR_LENGTH ,
" Invalid length " ) ;
}
2011-11-13 23:01:43 +04:00
if ( ! convert_string_talloc ( ndr - > current_mem_ctx , chset , CH_UNIX ,
ndr - > data + ndr - > offset , str_len * byte_mul ,
discard_const_p ( void * , var ) ,
& converted_size ) )
{
return ndr_pull_error ( ndr , NDR_ERR_CHARCNV ,
" Bad character conversion " ) ;
}
NDR_CHECK ( ndr_pull_advance ( ndr , length * byte_mul ) ) ;
return NDR_ERR_SUCCESS ;
}
2007-11-09 22:23:40 +03:00
_PUBLIC_ enum ndr_err_code ndr_push_charset ( struct ndr_push * ndr , int ndr_flags , const char * var , uint32_t length , uint8_t byte_mul , charset_t chset )
2006-07-11 22:01:26 +04:00
{
2018-03-26 13:00:40 +03:00
size_t required ;
2006-07-11 22:01:26 +04:00
2006-09-21 22:37:09 +04:00
if ( NDR_BE ( ndr ) & & chset = = CH_UTF16 ) {
chset = CH_UTF16BE ;
}
2006-07-11 22:01:26 +04:00
2018-03-26 13:02:01 +03:00
if ( ( byte_mul ! = 0 ) & & ( length > SIZE_MAX / byte_mul ) ) {
return ndr_push_error ( ndr , NDR_ERR_LENGTH , " length overflow " ) ;
}
2006-07-11 22:01:26 +04:00
required = byte_mul * length ;
NDR_PUSH_NEED_BYTES ( ndr , required ) ;
2011-03-30 20:58:22 +04:00
if ( required ) {
2011-03-31 00:08:31 +04:00
size_t size = 0 ;
2017-01-11 18:57:49 +03:00
if ( var = = NULL ) {
return ndr_push_error ( ndr , NDR_ERR_INVALID_POINTER , " NULL [ref] pointer " ) ;
}
2011-03-31 00:08:31 +04:00
if ( ! convert_string ( CH_UNIX , chset ,
2006-07-11 22:01:26 +04:00
var , strlen ( var ) ,
2011-03-31 00:08:31 +04:00
ndr - > data + ndr - > offset , required , & size ) ) {
2011-03-30 20:58:22 +04:00
return ndr_push_error ( ndr , NDR_ERR_CHARCNV ,
2008-10-13 17:33:40 +04:00
" Bad character conversion " ) ;
2011-03-30 20:58:22 +04:00
}
2006-07-11 22:01:26 +04:00
2011-03-30 20:58:22 +04:00
/* Make sure the remaining part of the string is filled with zeroes */
2011-03-31 00:08:31 +04:00
if ( size < required ) {
memset ( ndr - > data + ndr - > offset + size , 0 , required - size ) ;
2011-03-30 20:58:22 +04:00
}
2006-07-11 22:01:26 +04:00
}
ndr - > offset + = required ;
2007-11-09 22:23:40 +03:00
return NDR_ERR_SUCCESS ;
2006-07-11 22:01:26 +04:00
}
2017-01-11 18:57:49 +03:00
_PUBLIC_ enum ndr_err_code ndr_push_charset_to_null ( struct ndr_push * ndr , int ndr_flags , const char * var , uint32_t length , uint8_t byte_mul , charset_t chset )
{
const char * str = var ;
if ( str = = NULL ) {
2019-05-08 05:09:56 +03:00
str = " \0 " ; /* i.e. two zero bytes, for UTF16 null word. */
2017-05-08 12:22:51 +03:00
length = 1 ;
2017-01-11 18:57:49 +03:00
}
return ndr_push_charset ( ndr , ndr_flags , str , length , byte_mul , chset ) ;
}
2006-07-11 22:01:26 +04:00
/* Return number of elements in a string in the specified charset */
2007-11-08 16:36:13 +03:00
_PUBLIC_ uint32_t ndr_charset_length ( const void * var , charset_t chset )
2006-07-11 22:01:26 +04:00
{
2010-08-25 12:05:15 +04:00
switch ( chset ) {
/* case CH_UTF16: this has the same value as CH_UTF16LE */
case CH_UTF16LE :
case CH_UTF16BE :
case CH_UTF16MUNGED :
2010-10-31 03:04:25 +03:00
case CH_UTF8 :
return strlen_m_ext_term ( ( const char * ) var , CH_UNIX , chset ) ;
2010-08-25 12:05:15 +04:00
case CH_DOS :
case CH_UNIX :
return strlen ( ( const char * ) var ) + 1 ;
}
/* Fallback, this should never happen */
2006-07-11 22:01:26 +04:00
return strlen ( ( const char * ) var ) + 1 ;
}