Logo AND Algorithmique Numérique Distribuée

Public GIT Repository
ddcb5ba063ab496483e46149569156fb8b6da217
[simgrid.git] / src / xbt / dict_multi.c
1 /* $Id$ */
2
3 /* dict_multi - dictionnaries of dictionnaries of ... of data               */
4
5 /* Copyright (c) 2003-2005 Martin Quinson. All rights reserved.             */
6
7 /* This program is free software; you can redistribute it and/or modify it
8  * under the terms of the license (GNU LGPL) which comes with this package. */
9
10 #include "dict_private.h"
11
12 XBT_LOG_NEW_DEFAULT_SUBCATEGORY(dict_multi,dict, "Dictionaries of multiple keys");
13
14 static void _free_dict(void*d) {
15   xbt_dict_t dict=*(xbt_dict_t*)d;
16   xbt_dict_free(&dict);
17 }
18
19 /** \brief Insert \e data under all the keys contained in \e keys, providing their sizes in \e lens.
20  *
21  * \arg mdict: the multi-dict
22  * \arg keys: dynar of (char *) containing all the keys
23  * \arg lens: length of each element of \e keys
24  * \arg data: what to store in the structure
25  * \arg free_ctn: function to use to free the pushed content on need
26  *
27  * Dynars are not modified during the operation.
28  */
29
30 xbt_error_t
31 xbt_multidict_set_ext(xbt_dict_t  mdict,
32                       xbt_dynar_t keys, xbt_dynar_t     lens,
33                       void       *data, void_f_pvoid_t *free_ctn) {
34                       
35   xbt_error_t errcode;
36   xbt_dict_t thislevel,nextlevel;
37   int i;
38   
39   unsigned long int thislen;
40   char *thiskey;
41   int keys_len=xbt_dynar_length(keys);
42
43   xbt_assert(xbt_dynar_length(keys) == xbt_dynar_length(lens));
44   xbt_assert0(keys_len, "Can't set a zero-long key set in a multidict");
45
46   DEBUG2("xbt_multidict_set(%p,%d)", mdict, keys_len);
47
48   for (i=0         , thislevel = mdict    ; 
49        i<keys_len-1                       ; 
50        i++         , thislevel = nextlevel) {
51        
52     xbt_dynar_get_cpy(keys, i, &thiskey);
53     xbt_dynar_get_cpy(lens, i, &thislen);
54     
55     DEBUG5("multi_set: at level %d, len=%ld, key=%p |%*s|", i, thislen, thiskey, (int)thislen,thiskey);
56
57     /* search the dict of next level */
58     TRYCATCH(xbt_dict_get_ext(thislevel, thiskey, thislen, (void*)&nextlevel), mismatch_error);
59
60     /* make sure the dict of next level exists */
61     if (errcode == mismatch_error) {
62       nextlevel=xbt_dict_new();
63       xbt_dict_set_ext(thislevel, thiskey, thislen, &nextlevel, _free_dict);
64     }
65   }
66
67   xbt_dynar_get_cpy(keys, i, &thiskey);
68   xbt_dynar_get_cpy(lens, i, &thislen);
69   
70   xbt_dict_set_ext(thislevel, thiskey, thislen, data, free_ctn);
71   
72   return no_error;
73 }
74
75 /** \brief Insert \e data under all the keys contained in \e keys
76  *
77  * \arg head: the head of dict
78  * \arg keys: dynar of null-terminated strings containing all the keys
79  * \arg data: what to store in the structure
80  * \arg free_ctn: function to use to free the pushed content on need
81  */
82 xbt_error_t
83 xbt_multidict_set(xbt_dict_t  mdict,
84                   xbt_dynar_t keys,
85                   void       *data,  void_f_pvoid_t *free_ctn) {
86   xbt_error_t errcode;
87   xbt_dynar_t lens = xbt_dynar_new(sizeof(unsigned long int),NULL);
88   int i;
89
90   for (i = 0; i < xbt_dynar_length(keys); i++) {
91     char *thiskey = xbt_dynar_get_as(keys, i, char*);
92     unsigned long int thislen = (unsigned long int) strlen(thiskey);
93     DEBUG2("Push %ld as level %d length",thislen, i);
94     xbt_dynar_push(lens,&thislen);
95   }
96
97   errcode = xbt_multidict_set_ext(mdict, keys, lens, data, free_ctn);
98   xbt_dynar_free(&lens);         
99   return errcode;
100 }
101
102 /** \brief Insert \e data under all the keys contained in \e keys, providing their sizes in \e lens.
103  *
104  * \arg mdict: the multi-dict
105  * \arg keys: dynar of (char *) containing all the keys
106  * \arg lens: length of each element of \e keys
107  * \arg data: where to put what was found in structure
108  * \arg free_ctn: function to use to free the pushed content on need
109  *
110  * Dynars are not modified during the operation.
111  */
112 xbt_error_t
113 xbt_multidict_get_ext(xbt_dict_t  mdict,
114                       xbt_dynar_t keys,   xbt_dynar_t lens,
115                       /*OUT*/void **data) {
116   xbt_error_t errcode;
117   xbt_dict_t thislevel,nextlevel;
118   int i;
119
120   unsigned long int thislen;
121   char *thiskey;
122   int keys_len=xbt_dynar_length(keys);
123
124   xbt_assert(xbt_dynar_length(keys) == xbt_dynar_length(lens));
125   xbt_assert0(xbt_dynar_length(keys) >= 1, "Can't get a zero-long key set in a multidict");
126   
127   DEBUG2("xbt_multidict_get(%p, %ld)", mdict, xbt_dynar_length(keys));
128
129   for (i=0         , thislevel=mdict      ; 
130        i<keys_len-1                       ; 
131        i++         , thislevel = nextlevel) {
132        
133     xbt_dynar_get_cpy(keys, i, &thiskey);
134     xbt_dynar_get_cpy(lens, i, &thislen);
135
136     DEBUG5("multi_get: at level %d, len=%ld, key=%p |%*s|", i, thislen, thiskey, (int)thislen,thiskey);
137
138     /* search the dict of next level: let mismatch raise if not found */
139     TRY(xbt_dict_get_ext(thislevel, thiskey, thislen, (void*)&nextlevel));
140   }
141   
142   xbt_dynar_get_cpy(keys, i, &thiskey);
143   xbt_dynar_get_cpy(lens, i, &thislen);
144   
145   return xbt_dict_get_ext(thislevel, thiskey, thislen, data);
146 }
147
148 xbt_error_t
149 xbt_multidict_get(xbt_dict_t mdict, xbt_dynar_t keys, /*OUT*/void **data) {
150   xbt_error_t errcode;
151   xbt_dynar_t lens = xbt_dynar_new(sizeof(unsigned long int),NULL);
152   int i;
153   
154   for (i = 0; i < xbt_dynar_length(keys); i++) {
155     char *thiskey = xbt_dynar_get_as(keys, i, char*);
156     unsigned long int thislen = (unsigned long int) strlen(thiskey);
157     xbt_dynar_push(lens,&thislen);
158   }
159
160   errcode = xbt_multidict_get_ext(mdict, keys, lens, data),
161   xbt_dynar_free(&lens);         
162   return errcode;
163 }
164
165
166 /** \brief Remove the entry under all the keys contained in \e keys, providing their sizes in \e lens.
167  *
168  * \arg mdict: the multi-dict
169  * \arg keys: dynar of (char *) containing all the keys
170  * \arg lens: length of each element of \e keys
171  * \arg data: what to store in the structure
172  * \arg free_ctn: function to use to free the pushed content on need
173  *
174  * Dynars are not modified during the operation.
175  *
176  * Removing a non-existant key is ok.
177  */
178
179 xbt_error_t
180 xbt_multidict_remove_ext(xbt_dict_t mdict, xbt_dynar_t keys, xbt_dynar_t lens) {
181   xbt_error_t errcode;
182   xbt_dict_t thislevel,nextlevel;
183   int i;
184
185   xbt_assert(xbt_dynar_length(keys) == xbt_dynar_length(lens));
186   xbt_assert0(xbt_dynar_length(keys), "Can't remove a zero-long key set in a multidict");
187
188   unsigned long int thislen;
189   char *thiskey;
190   int keys_len=xbt_dynar_length(keys);
191
192   xbt_assert(xbt_dynar_length(keys) == xbt_dynar_length(lens));
193   xbt_assert0(keys_len, "Can't set a zero-long key set in a multidict");
194
195   for (i=0         , thislevel=mdict      ; 
196        i<keys_len-1                       ; 
197        i++         , thislevel = nextlevel) {
198        
199     xbt_dynar_get_cpy(keys, i, &thiskey);
200     xbt_dynar_get_cpy(lens, i, &thislen);
201
202     /* search the dict of next level */
203     TRYCATCH(xbt_dict_get_ext(thislevel, thiskey, thislen, (void*)&nextlevel), mismatch_error);
204
205     /* If non-existant entry, nothing to do */
206     if (errcode == mismatch_error)
207       return no_error;
208   }
209
210   xbt_dynar_get_cpy(keys, i, &thiskey);
211   xbt_dynar_get_cpy(lens, i, &thislen);
212   
213   return xbt_dict_remove_ext(thislevel, thiskey, thislen);
214 }
215
216 xbt_error_t
217 xbt_multidict_remove(xbt_dict_t mdict, xbt_dynar_t keys) {
218
219   xbt_error_t errcode;
220   xbt_dynar_t lens = xbt_dynar_new(sizeof(unsigned long int),NULL);
221   int i;
222       
223   for (i = 0; i < xbt_dynar_length(keys); i++) {
224     char *thiskey = xbt_dynar_get_as(keys, i, char*);
225     unsigned long int thislen = strlen(thiskey);
226     xbt_dynar_push(lens,&thislen);
227   }
228                       
229   errcode = xbt_multidict_remove_ext(mdict, keys, lens);
230   xbt_dynar_free(&lens);
231   return errcode;
232 }