2003-08-13 05:53:07 +04:00
/*
Unix SMB / CIFS implementation .
Easy management of byte - length data
Copyright ( C ) Andrew Tridgell 2001
Copyright ( C ) Andrew Bartlett 2001
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 2 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 , write to the Free Software
Foundation , Inc . , 675 Mass Ave , Cambridge , MA 0213 9 , USA .
*/
# include "includes.h"
2006-02-28 16:12:39 +03:00
/**
* @ file
* @ brief Manipulation of arbitrary data blobs
* */
/**
2003-08-13 05:53:07 +04:00
construct a data blob , must be freed with data_blob_free ( )
you can pass NULL for p and get a blank data blob
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ DATA_BLOB data_blob_named ( const void * p , size_t length , const char * name )
2003-08-13 05:53:07 +04:00
{
DATA_BLOB ret ;
r6528: - in tdb_fetch() we effectively disallowed zero length records by
returning NULL/0, which is the same as we used for a failure. Having
to look at tdb->ecode (which we never do) is too error prone.
Instead, tdb_fetch() should behave like malloc() and talloc(), where
zero length is not special and malloc(0) returns a valid pointer.
- similarly in data_blob(), asking for data_blob(NULL, 0) should
return a zero blob, but asking for data_blob(ptr, 0) should return a
zero length blob with a valid pointer, just like talloc() and malloc()
This change fixes the SummaryInformation stream stored in the tdb
backend when manipulated from w2k. The w2k client was using
SET_EOF_INFORMATION to create a zero-length stream, which we return
STATUS_NOT_FOUND on, as the tdb_fetch() gave us back a NULL/0 blob,
which we returned as not-found
(This used to be commit 162bbe4402b9de6ac06103df904b9fc204fbff29)
2005-04-30 13:04:14 +04:00
if ( p = = NULL & & length = = 0 ) {
2003-08-13 05:53:07 +04:00
ZERO_STRUCT ( ret ) ;
return ret ;
}
if ( p ) {
2004-08-25 07:23:39 +04:00
ret . data = talloc_memdup ( NULL , p , length ) ;
2003-08-13 05:53:07 +04:00
} else {
2005-01-06 06:06:58 +03:00
ret . data = talloc_size ( NULL , length ) ;
2004-08-25 07:23:39 +04:00
}
if ( ret . data = = NULL ) {
ret . length = 0 ;
return ret ;
2003-08-13 05:53:07 +04:00
}
2004-09-26 10:41:59 +04:00
talloc_set_name_const ( ret . data , name ) ;
2003-08-13 05:53:07 +04:00
ret . length = length ;
return ret ;
}
2006-02-28 16:12:39 +03:00
/**
2003-08-13 05:53:07 +04:00
construct a data blob , using supplied TALLOC_CTX
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ DATA_BLOB data_blob_talloc_named ( TALLOC_CTX * mem_ctx , const void * p , size_t length , const char * name )
2003-08-13 05:53:07 +04:00
{
2004-09-27 08:20:18 +04:00
DATA_BLOB ret = data_blob_named ( p , length , name ) ;
2003-08-13 05:53:07 +04:00
2004-08-25 07:23:39 +04:00
if ( ret . data ) {
2004-09-27 08:20:18 +04:00
talloc_steal ( mem_ctx , ret . data ) ;
2003-08-13 05:53:07 +04:00
}
return ret ;
}
2004-11-06 06:44:16 +03:00
2006-02-28 16:12:39 +03:00
/**
2004-11-06 08:40:34 +03:00
reference a data blob , to the supplied TALLOC_CTX .
Returns a NULL DATA_BLOB on failure
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ DATA_BLOB data_blob_talloc_reference ( TALLOC_CTX * mem_ctx , DATA_BLOB * blob )
2004-11-06 06:44:16 +03:00
{
DATA_BLOB ret = * blob ;
2004-11-06 08:40:34 +03:00
ret . data = talloc_reference ( mem_ctx , blob - > data ) ;
if ( ! ret . data ) {
return data_blob ( NULL , 0 ) ;
}
2004-11-06 06:44:16 +03:00
return ret ;
}
2006-02-28 16:12:39 +03:00
/**
2003-12-13 05:20:40 +03:00
construct a zero data blob , using supplied TALLOC_CTX .
use this sparingly as it initialises data - better to initialise
yourself if you want specific data in the blob
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ DATA_BLOB data_blob_talloc_zero ( TALLOC_CTX * mem_ctx , size_t length )
2003-12-13 05:20:40 +03:00
{
DATA_BLOB blob = data_blob_talloc ( mem_ctx , NULL , length ) ;
data_blob_clear ( & blob ) ;
return blob ;
}
2006-02-28 16:12:39 +03:00
/**
2003-08-13 05:53:07 +04:00
free a data blob
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ void data_blob_free ( DATA_BLOB * d )
2003-08-13 05:53:07 +04:00
{
if ( d ) {
2004-08-25 07:23:39 +04:00
talloc_free ( d - > data ) ;
2004-05-09 16:30:33 +04:00
d - > data = NULL ;
2003-08-13 05:53:07 +04:00
d - > length = 0 ;
}
}
2006-02-28 16:12:39 +03:00
/**
2003-08-13 05:53:07 +04:00
clear a DATA_BLOB ' s contents
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ void data_blob_clear ( DATA_BLOB * d )
2003-08-13 05:53:07 +04:00
{
if ( d - > data ) {
memset ( d - > data , 0 , d - > length ) ;
}
}
2006-02-28 16:12:39 +03:00
/**
2003-08-13 05:53:07 +04:00
free a data blob and clear its contents
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ void data_blob_clear_free ( DATA_BLOB * d )
2003-08-13 05:53:07 +04:00
{
data_blob_clear ( d ) ;
data_blob_free ( d ) ;
}
2006-02-28 16:12:39 +03:00
/**
2003-08-13 05:53:07 +04:00
check if two data blobs are equal
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ BOOL data_blob_equal ( const DATA_BLOB * d1 , const DATA_BLOB * d2 )
2003-08-13 05:53:07 +04:00
{
if ( d1 - > length ! = d2 - > length ) {
return False ;
}
if ( d1 - > data = = d2 - > data ) {
return True ;
}
if ( d1 - > data = = NULL | | d2 - > data = = NULL ) {
return False ;
}
if ( memcmp ( d1 - > data , d2 - > data , d1 - > length ) = = 0 ) {
return True ;
}
return False ;
}
2006-02-28 16:12:39 +03:00
/**
2004-12-01 19:51:37 +03:00
print the data_blob as hex string
2006-02-28 16:12:39 +03:00
* */
2007-01-13 18:01:39 +03:00
_PUBLIC_ char * data_blob_hex_string ( TALLOC_CTX * mem_ctx , const DATA_BLOB * blob )
2004-12-01 19:51:37 +03:00
{
int i ;
char * hex_string ;
2005-01-27 10:08:20 +03:00
hex_string = talloc_array ( mem_ctx , char , ( blob - > length * 2 ) + 1 ) ;
2004-12-01 19:51:37 +03:00
if ( ! hex_string ) {
return NULL ;
}
for ( i = 0 ; i < blob - > length ; i + + )
slprintf ( & hex_string [ i * 2 ] , 3 , " %02X " , blob - > data [ i ] ) ;
2007-01-13 18:39:49 +03:00
hex_string [ ( blob - > length * 2 ) ] = ' \0 ' ;
2004-12-01 19:51:37 +03:00
return hex_string ;
}
2004-12-18 07:38:43 +03:00
2006-02-28 16:12:39 +03:00
/**
2004-12-18 07:38:43 +03:00
useful for constructing data blobs in test suites , while
avoiding const warnings
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ DATA_BLOB data_blob_string_const ( const char * str )
2004-12-18 07:38:43 +03:00
{
DATA_BLOB blob ;
blob . data = discard_const ( str ) ;
blob . length = strlen ( str ) ;
return blob ;
}
2006-03-06 03:24:51 +03:00
/**
* Create a new data blob from const data
*/
2006-03-05 20:15:19 +03:00
_PUBLIC_ DATA_BLOB data_blob_const ( const void * p , size_t length )
2004-12-21 18:15:49 +03:00
{
DATA_BLOB blob ;
blob . data = discard_const ( p ) ;
blob . length = length ;
return blob ;
}
2005-05-26 05:00:58 +04:00
2006-02-28 16:12:39 +03:00
/**
2005-11-09 11:06:56 +03:00
realloc a data_blob
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ NTSTATUS data_blob_realloc ( TALLOC_CTX * mem_ctx , DATA_BLOB * blob , size_t length )
2005-11-09 11:06:56 +03:00
{
blob - > data = talloc_realloc_size ( mem_ctx , blob - > data , length ) ;
NT_STATUS_HAVE_NO_MEMORY ( blob - > data ) ;
blob - > length = length ;
return NT_STATUS_OK ;
}
2006-07-25 04:53:03 +04:00
2006-02-28 16:12:39 +03:00
/**
2005-05-26 05:00:58 +04:00
append some data to a data blob
2006-02-28 16:12:39 +03:00
* */
2006-03-05 20:15:19 +03:00
_PUBLIC_ NTSTATUS data_blob_append ( TALLOC_CTX * mem_ctx , DATA_BLOB * blob ,
2006-07-21 04:56:48 +04:00
const void * p , size_t length )
2005-05-26 05:00:58 +04:00
{
2006-07-25 04:53:03 +04:00
NTSTATUS status ;
size_t old_len = blob - > length ;
size_t new_len = old_len + length ;
if ( new_len < length | | new_len < old_len ) {
return NT_STATUS_NO_MEMORY ;
}
if ( ( const uint8_t * ) p + length < ( const uint8_t * ) p ) {
return NT_STATUS_NO_MEMORY ;
}
status = data_blob_realloc ( mem_ctx , blob , new_len ) ;
if ( ! NT_STATUS_IS_OK ( status ) ) {
return status ;
}
memcpy ( blob - > data + old_len , p , length ) ;
2005-05-26 05:00:58 +04:00
return NT_STATUS_OK ;
}
2005-11-09 11:06:56 +03:00