2002-07-18 23:00:24 +00:00
/*
* Unix SMB / CIFS implementation .
* Generic Abstract Data Types
* Copyright ( C ) Gerald Carter 2002.
*
* 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 19:25:36 +00:00
* the Free Software Foundation ; either version 3 of the License , or
2002-07-18 23:00:24 +00: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 05:23:25 +00:00
* along with this program ; if not , see < http : //www.gnu.org/licenses/>.
2002-07-18 23:00:24 +00:00
*/
# include "includes.h"
2005-02-23 16:36:44 +00:00
# include "adt_tree.h"
2002-07-18 23:00:24 +00:00
2010-02-07 15:47:07 +01:00
struct tree_node {
struct tree_node * parent ;
struct tree_node * * children ;
int num_children ;
char * key ;
void * data_p ;
} ;
struct sorted_tree {
struct tree_node * root ;
} ;
2002-07-18 23:00:24 +00:00
2002-07-20 02:42:04 +00:00
/**************************************************************************
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-10-18 17:40:25 -07:00
static bool trim_tree_keypath ( char * path , char * * base , char * * new_path )
2002-07-20 02:42:04 +00:00
{
char * p ;
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
* new_path = * base = NULL ;
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
if ( ! path )
return False ;
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
* base = path ;
2010-02-07 15:01:57 +01:00
2010-06-24 16:33:37 +02:00
p = strchr ( path , ' \\ ' ) ;
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
if ( p ) {
* p = ' \0 ' ;
* new_path = p + 1 ;
}
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
return True ;
}
2002-07-18 23:00:24 +00:00
/**************************************************************************
2010-02-07 15:49:13 +01:00
Initialize the tree ' s root .
2002-07-18 23:00:24 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-02-07 15:49:13 +01:00
struct sorted_tree * pathtree_init ( void * data_p )
2002-07-18 23:00:24 +00:00
{
2010-02-07 15:45:42 +01:00
struct sorted_tree * tree = NULL ;
2010-02-07 15:01:57 +01:00
2010-02-07 15:45:42 +01:00
tree = talloc_zero ( NULL , struct sorted_tree ) ;
if ( tree = = NULL ) {
2002-07-18 23:00:24 +00:00
return NULL ;
2010-02-07 15:45:42 +01:00
}
2010-02-07 15:01:57 +01:00
2010-02-07 15:42:26 +01:00
tree - > root = talloc_zero ( tree , struct tree_node ) ;
if ( tree - > root = = NULL ) {
2005-09-03 16:38:51 +00:00
TALLOC_FREE ( tree ) ;
2002-07-18 23:00:24 +00:00
return NULL ;
}
2010-02-07 15:01:57 +01:00
2002-07-19 18:49:44 +00:00
tree - > root - > data_p = data_p ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
return tree ;
}
/**************************************************************************
Find the next child given a key string
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-02-07 15:42:26 +01:00
static struct tree_node * pathtree_birth_child ( struct tree_node * node ,
char * key )
2002-07-18 23:00:24 +00:00
{
2010-02-07 15:42:26 +01:00
struct tree_node * infant = NULL ;
struct tree_node * * siblings ;
2002-07-20 02:42:04 +00:00
int i ;
2010-02-07 15:01:57 +01:00
2010-02-07 15:42:26 +01:00
infant = talloc_zero ( node , struct tree_node ) ;
if ( infant = = NULL ) {
2002-07-18 23:00:24 +00:00
return NULL ;
2010-02-07 15:42:26 +01:00
}
2010-02-07 15:01:57 +01:00
2005-09-03 16:38:51 +00:00
infant - > key = talloc_strdup ( infant , key ) ;
2002-07-18 23:00:24 +00:00
infant - > parent = node ;
2010-02-07 15:01:57 +01:00
2010-02-07 15:42:26 +01:00
siblings = talloc_realloc ( node , node - > children , struct tree_node * ,
node - > num_children + 1 ) ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( siblings )
node - > children = siblings ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
node - > num_children + + ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
/* first child */
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( node - > num_children = = 1 ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 11 , ( " pathtree_birth_child: First child of node [%s]! [%s] \n " ,
2002-07-18 23:00:24 +00:00
node - > key ? node - > key : " NULL " , infant - > key ) ) ;
node - > children [ 0 ] = infant ;
}
else
{
/*
* multiple siblings . . . . ( at least 2 children )
*
* work from the end of the list forward
* The last child is not set at this point
* Insert the new infanct in ascending order
* from left to right
*/
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
for ( i = node - > num_children - 1 ; i > = 1 ; i - - )
{
2005-02-23 16:36:44 +00:00
DEBUG ( 11 , ( " pathtree_birth_child: Looking for crib; infant -> [%s], child -> [%s] \n " ,
2002-07-19 18:49:44 +00:00
infant - > key , node - > children [ i - 1 ] - > key ) ) ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
/* the strings should never match assuming that we
2005-02-23 16:36:44 +00:00
have called pathtree_find_child ( ) first */
2010-02-07 15:01:57 +01:00
2011-05-13 20:21:30 +02:00
if ( strcasecmp_m ( infant - > key , node - > children [ i - 1 ] - > key ) > 0 ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 11 , ( " pathtree_birth_child: storing infant in i == [%d] \n " ,
2002-07-20 02:42:04 +00:00
i ) ) ;
2002-07-19 18:49:44 +00:00
node - > children [ i ] = infant ;
2002-07-18 23:00:24 +00:00
break ;
}
2010-02-07 15:01:57 +01:00
2002-07-19 18:49:44 +00:00
/* bump everything towards the end on slot */
2010-02-07 15:01:57 +01:00
2002-07-19 18:49:44 +00:00
node - > children [ i ] = node - > children [ i - 1 ] ;
2002-07-18 23:00:24 +00:00
}
2002-07-20 02:42:04 +00:00
2005-02-23 16:36:44 +00:00
DEBUG ( 11 , ( " pathtree_birth_child: Exiting loop (i == [%d]) \n " , i ) ) ;
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
/* if we haven't found the correct slot yet, the child
2002-07-19 18:49:44 +00:00
will be first in the list */
2010-02-07 15:01:57 +01:00
2002-07-19 18:49:44 +00:00
if ( i = = 0 )
node - > children [ 0 ] = infant ;
2002-07-18 23:00:24 +00:00
}
return infant ;
}
2002-07-20 02:42:04 +00:00
2002-07-18 23:00:24 +00:00
/**************************************************************************
Find the next child given a key string
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-02-07 15:42:26 +01:00
static struct tree_node * pathtree_find_child ( struct tree_node * node ,
char * key )
2002-07-18 23:00:24 +00:00
{
2010-02-07 15:42:26 +01:00
struct tree_node * next = NULL ;
2002-07-18 23:00:24 +00:00
int i , result ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( ! node ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_find_child: NULL node passed into function! \n " ) ) ;
2002-07-18 23:00:24 +00:00
return NULL ;
}
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( ! key ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_find_child: NULL key string passed into function! \n " ) ) ;
2002-07-18 23:00:24 +00:00
return NULL ;
}
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
for ( i = 0 ; i < node - > num_children ; i + + )
{
2005-02-23 16:36:44 +00:00
DEBUG ( 11 , ( " pathtree_find_child: child key => [%s] \n " ,
2002-07-20 02:42:04 +00:00
node - > children [ i ] - > key ) ) ;
2010-02-07 15:01:57 +01:00
2011-05-13 20:21:30 +02:00
result = strcasecmp_m ( node - > children [ i ] - > key , key ) ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( result = = 0 )
next = node - > children [ i ] ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
/* if result > 0 then we've gone to far because
the list of children is sorted by key name
If result = = 0 , then we have a match */
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
if ( result > 0 )
2002-07-18 23:00:24 +00:00
break ;
}
2005-02-23 16:36:44 +00:00
DEBUG ( 11 , ( " pathtree_find_child: %s [%s] \n " ,
2002-07-20 02:42:04 +00:00
next ? " Found " : " Did not find " , key ) ) ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
return next ;
}
/**************************************************************************
Add a new node into the tree given a key path and a blob of data
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2012-03-24 16:41:35 +01:00
bool pathtree_add ( struct sorted_tree * tree , const char * path , void * data_p )
2002-07-18 23:00:24 +00:00
{
char * str , * base , * path2 ;
2010-02-07 15:42:26 +01:00
struct tree_node * current , * next ;
2012-03-24 16:41:35 +01:00
bool ret = true ;
2010-02-07 15:01:57 +01:00
2005-02-23 16:36:44 +00:00
DEBUG ( 8 , ( " pathtree_add: Enter \n " ) ) ;
2010-02-07 15:01:57 +01:00
2010-06-24 16:33:37 +02:00
if ( ! path | | * path ! = ' \\ ' ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_add: Attempt to add a node with a bad path [%s] \n " ,
2002-07-18 23:00:24 +00:00
path ? path : " NULL " ) ) ;
2012-03-24 17:11:11 +01:00
return false ;
2002-07-18 23:00:24 +00:00
}
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( ! tree ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_add: Attempt to add a node to an uninitialized tree! \n " ) ) ;
2012-03-24 17:11:11 +01:00
return false ;
2002-07-18 23:00:24 +00:00
}
2010-02-07 15:01:57 +01:00
2010-06-24 16:33:37 +02:00
/* move past the first '\\' */
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
path + + ;
2004-12-07 18:25:53 +00:00
path2 = SMB_STRDUP ( path ) ;
2002-07-18 23:00:24 +00:00
if ( ! path2 ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_add: strdup() failed on string [%s]!?!?! \n " , path ) ) ;
2012-03-24 17:11:11 +01:00
return false ;
2002-07-18 23:00:24 +00:00
}
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
/*
* this works sort of like a ' mkdir - p ' call , possibly
* creating an entire path to the new node at once
* The path should be of the form / < key1 > / < key2 > / . . .
*/
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
base = path2 ;
str = path2 ;
current = tree - > root ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
do {
/* break off the remaining part of the path */
2010-02-07 15:01:57 +01:00
2010-06-24 16:33:37 +02:00
str = strchr ( str , ' \\ ' ) ;
2002-07-18 23:00:24 +00:00
if ( str )
* str = ' \0 ' ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
/* iterate to the next child--birth it if necessary */
2010-02-07 15:01:57 +01:00
2005-02-23 16:36:44 +00:00
next = pathtree_find_child ( current , base ) ;
2002-07-18 23:00:24 +00:00
if ( ! next ) {
2005-02-23 16:36:44 +00:00
next = pathtree_birth_child ( current , base ) ;
2002-07-18 23:00:24 +00:00
if ( ! next ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_add: Failed to create new child! \n " ) ) ;
2012-03-24 16:41:35 +01:00
ret = false ;
2002-07-18 23:00:24 +00:00
goto done ;
}
}
current = next ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
/* setup the next part of the path */
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
base = str ;
if ( base ) {
2010-06-24 16:33:37 +02:00
* base = ' \\ ' ;
2002-07-18 23:00:24 +00:00
base + + ;
str = base ;
}
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
} while ( base ! = NULL ) ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
current - > data_p = data_p ;
2010-02-07 15:01:57 +01:00
2005-02-23 16:36:44 +00:00
DEBUG ( 10 , ( " pathtree_add: Successfully added node [%s] to tree \n " ,
2002-07-18 23:00:24 +00:00
path ) ) ;
2005-02-23 16:36:44 +00:00
DEBUG ( 8 , ( " pathtree_add: Exit \n " ) ) ;
2002-07-18 23:00:24 +00:00
done :
SAFE_FREE ( path2 ) ;
return ret ;
}
/**************************************************************************
2010-02-07 15:42:26 +01:00
Recursive routine to print out all children of a struct tree_node
2002-07-18 23:00:24 +00:00
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2007-09-14 22:14:39 +00:00
static void pathtree_print_children ( TALLOC_CTX * ctx ,
2010-02-07 15:42:26 +01:00
struct tree_node * node ,
2007-09-14 22:14:39 +00:00
int debug ,
const char * path )
2002-07-18 23:00:24 +00:00
{
int i ;
int num_children ;
2007-09-14 22:14:39 +00:00
char * path2 = NULL ;
2002-07-18 23:00:24 +00:00
if ( ! node )
return ;
2007-09-14 22:14:39 +00:00
2002-07-18 23:00:24 +00:00
if ( node - > key )
DEBUG ( debug , ( " %s: [%s] (%s) \n " , path ? path : " NULL " , node - > key ,
node - > data_p ? " data " : " NULL " ) ) ;
2007-09-14 22:14:39 +00:00
if ( path ) {
path2 = talloc_strdup ( ctx , path ) ;
if ( ! path2 ) {
return ;
}
}
path2 = talloc_asprintf ( ctx ,
" %s%s/ " ,
path ? path : " " ,
node - > key ? node - > key : " NULL " ) ;
if ( ! path2 ) {
return ;
}
2002-07-18 23:00:24 +00:00
2007-09-14 22:14:39 +00:00
num_children = node - > num_children ;
for ( i = 0 ; i < num_children ; i + + ) {
pathtree_print_children ( ctx , node - > children [ i ] , debug , path2 ) ;
}
2002-07-18 23:00:24 +00:00
}
/**************************************************************************
Dump the kys for a tree to the log file
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-02-07 15:45:42 +01:00
void pathtree_print_keys ( struct sorted_tree * tree , int debug )
2002-07-18 23:00:24 +00:00
{
int i ;
int num_children = tree - > root - > num_children ;
2007-09-14 22:14:39 +00:00
2002-07-18 23:00:24 +00:00
if ( tree - > root - > key )
DEBUG ( debug , ( " ROOT/: [%s] (%s) \n " , tree - > root - > key ,
tree - > root - > data_p ? " data " : " NULL " ) ) ;
2007-09-14 22:14:39 +00:00
2002-07-18 23:00:24 +00:00
for ( i = 0 ; i < num_children ; i + + ) {
2007-09-14 22:14:39 +00:00
TALLOC_CTX * ctx = talloc_stackframe ( ) ;
pathtree_print_children ( ctx , tree - > root - > children [ i ] , debug ,
2002-07-18 23:00:24 +00:00
tree - > root - > key ? tree - > root - > key : " ROOT/ " ) ;
2007-09-14 22:14:39 +00:00
TALLOC_FREE ( ctx ) ;
2002-07-18 23:00:24 +00:00
}
2007-09-14 22:14:39 +00:00
2002-07-18 23:00:24 +00:00
}
/**************************************************************************
return the data_p for for the node in tree matching the key string
2007-09-14 22:14:39 +00:00
The key string is the full path . We must break it apart and walk
2002-07-18 23:00:24 +00:00
the tree
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2010-02-07 15:45:42 +01:00
void * pathtree_find ( struct sorted_tree * tree , char * key )
2002-07-18 23:00:24 +00:00
{
2007-01-16 18:05:37 +00:00
char * keystr , * base = NULL , * str = NULL , * p ;
2010-02-07 15:42:26 +01:00
struct tree_node * current ;
2002-07-18 23:00:24 +00:00
void * result = NULL ;
2010-02-07 15:01:57 +01:00
2005-02-23 16:36:44 +00:00
DEBUG ( 10 , ( " pathtree_find: Enter [%s] \n " , key ? key : " NULL " ) ) ;
2002-07-18 23:00:24 +00:00
/* sanity checks first */
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( ! key ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_find: Attempt to search tree using NULL search string! \n " ) ) ;
2002-07-18 23:00:24 +00:00
return NULL ;
}
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( ! tree ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_find: Attempt to search an uninitialized tree using string [%s]! \n " ,
2002-07-18 23:00:24 +00:00
key ? key : " NULL " ) ) ;
return NULL ;
}
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( ! tree - > root )
return NULL ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
/* make a copy to play with */
2010-02-07 15:01:57 +01:00
2010-06-24 16:33:37 +02:00
if ( * key = = ' \\ ' )
2004-12-07 18:25:53 +00:00
keystr = SMB_STRDUP ( key + 1 ) ;
2002-07-20 02:42:04 +00:00
else
2004-12-07 18:25:53 +00:00
keystr = SMB_STRDUP ( key ) ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
if ( ! keystr ) {
2005-02-23 16:36:44 +00:00
DEBUG ( 0 , ( " pathtree_find: strdup() failed on string [%s]!?!?! \n " , key ) ) ;
2002-07-18 23:00:24 +00:00
return NULL ;
}
/* start breaking the path apart */
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
p = keystr ;
2002-07-18 23:00:24 +00:00
current = tree - > root ;
2010-02-07 15:01:57 +01:00
2002-07-19 18:49:44 +00:00
if ( tree - > root - > data_p )
result = tree - > root - > data_p ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
do
{
/* break off the remaining part of the path */
2002-07-20 02:42:04 +00:00
trim_tree_keypath ( p , & base , & str ) ;
2010-02-07 15:01:57 +01:00
2005-02-23 16:36:44 +00:00
DEBUG ( 11 , ( " pathtree_find: [loop] base => [%s], new_path => [%s] \n " ,
2007-01-16 18:05:37 +00:00
base ? base : " " ,
str ? str : " " ) ) ;
2002-07-18 23:00:24 +00:00
/* iterate to the next child */
2010-02-07 15:01:57 +01:00
2005-02-23 16:36:44 +00:00
current = pathtree_find_child ( current , base ) ;
2010-02-07 15:01:57 +01:00
2002-07-19 18:49:44 +00:00
/*
* the idea is that the data_p for a parent should
* be inherited by all children , but allow it to be
* overridden farther down
*/
2010-02-07 15:01:57 +01:00
2002-07-19 18:49:44 +00:00
if ( current & & current - > data_p )
result = current - > data_p ;
2002-07-20 02:42:04 +00:00
/* reset the path pointer 'p' to the remaining part of the key string */
p = str ;
2010-02-07 15:01:57 +01:00
2002-07-20 02:42:04 +00:00
} while ( str & & current ) ;
2010-02-07 15:01:57 +01:00
2002-07-19 18:49:44 +00:00
/* result should be the data_p from the lowest match node in the tree */
2002-07-19 22:16:03 +00:00
if ( result )
2005-02-23 16:36:44 +00:00
DEBUG ( 11 , ( " pathtree_find: Found data_p! \n " ) ) ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
SAFE_FREE ( keystr ) ;
2010-02-07 15:01:57 +01:00
2005-02-23 16:36:44 +00:00
DEBUG ( 10 , ( " pathtree_find: Exit \n " ) ) ;
2010-02-07 15:01:57 +01:00
2002-07-18 23:00:24 +00:00
return result ;
}