Logo AND Algorithmique Numérique Distribuée

Public GIT Repository
Merge branch 'master' of git+ssh://scm.gforge.inria.fr//gitroot/simgrid/simgrid
[simgrid.git] / src / xbt / lib.c
1 /* lib - a generic library, variation over dictionary                    */
2
3 /* Copyright (c) 2011, 2013-2015. The SimGrid Team.
4  * All rights reserved.                                                     */
5
6 /* This program is free software; you can redistribute it and/or modify it
7  * under the terms of the license (GNU LGPL) which comes with this package. */
8
9 #include <xbt/sysdep.h>
10 #include <xbt/log.h>
11 #include <xbt/lib.h>
12
13 XBT_LOG_NEW_DEFAULT_SUBCATEGORY(xbt_lib, xbt,
14                                 "A dict with keys of type (name, level)");
15
16 xbt_lib_t xbt_lib_new(void)
17 {
18   xbt_lib_t lib  = xbt_new(s_xbt_lib_t, 1);
19   lib->dict = xbt_dict_new_homogeneous(xbt_free_f);
20   lib->levels = 0;
21   lib->free_f = NULL;
22   return lib;
23 }
24
25 void xbt_lib_free(xbt_lib_t *plib)
26 {
27   xbt_lib_t lib = *plib;
28   if (lib) {
29     xbt_dict_cursor_t cursor;
30     char *key;
31     void **elts;
32     xbt_dict_foreach (lib->dict, cursor, key, elts) {
33       int i;
34       for (i = 0 ; i < lib->levels ; i++)
35         if (elts[i] && lib->free_f[i])
36           lib->free_f[i](elts[i]);
37     }
38     xbt_dict_free(&lib->dict);
39     xbt_free(lib->free_f);
40     xbt_free(lib);
41     *plib = NULL;
42   }
43 }
44
45 int xbt_lib_add_level(xbt_lib_t lib, void_f_pvoid_t free_f)
46 {
47   XBT_DEBUG("xbt_lib_add_level");
48   xbt_assert(xbt_dict_is_empty(lib->dict), "Lib is not empty, cannot add a level");
49   lib->free_f = xbt_realloc(lib->free_f, sizeof(void_f_pvoid_t) * (lib->levels + 1));
50   lib->free_f[lib->levels] = free_f;
51   return lib->levels++;
52 }
53
54 void xbt_lib_set(xbt_lib_t lib, const char *key, int level, void *obj)
55 {
56   XBT_DEBUG("xbt_lib_set key '%s:%d' with object %p", key, level, obj);
57   void **elts = xbt_dict_get_or_null(lib->dict, key);
58   if (!elts) {
59     elts = xbt_new0(void *, lib->levels);
60     xbt_dict_set(lib->dict, key, elts, NULL);
61   }
62   if (elts[level]) {
63     XBT_DEBUG("Replace %p by %p element under key '%s:%d'", elts[level], obj, key, level);
64     if (lib->free_f[level])
65       lib->free_f[level](elts[level]);
66   }
67   elts[level] = obj;
68 }
69
70 void xbt_lib_unset(xbt_lib_t lib, const char *key, int level, int invoke_callback)
71 {
72   void **elts = xbt_dict_get_or_null(lib->dict, key);
73   if (!elts) {
74      XBT_WARN("no key %s", key);
75      return;
76   }
77
78   void *obj = elts[level];
79   if (!obj) {
80      XBT_WARN("no key %s at level %d", key, level);
81      return;
82   }
83
84   XBT_DEBUG("Remove %p of key %s at level %d", obj, key, level);
85   elts[level] = NULL;
86
87   /* check if there still remains any elements of this key */
88   int empty = 1;
89   for (int i = 0; i < lib->levels && empty; i++) {
90      if (elts[i] != NULL)
91        empty = 0;
92   }
93   if (empty) {
94     /* there is no element at any level, so delete the key */
95     xbt_dict_remove(lib->dict, key);
96   }
97
98   if (invoke_callback && lib->free_f[level])
99     lib->free_f[level](obj);
100 }
101
102 void *xbt_lib_get_or_null(xbt_lib_t lib, const char *key, int level)
103 {
104   void **elts = xbt_dict_get_or_null(lib->dict, key);
105   return elts ? elts[level] : NULL;
106 }
107
108 xbt_dictelm_t xbt_lib_get_elm_or_null(xbt_lib_t lib, const char *key)
109 {
110   return xbt_dict_get_elm_or_null(lib->dict, key);
111 }
112
113 void *xbt_lib_get_level(xbt_dictelm_t elm, int level){
114   void **elts = elm->content;
115   return elts ? elts[level] : NULL;
116 }
117
118 void xbt_lib_remove(xbt_lib_t lib, const char *key){
119   xbt_dict_remove(lib->dict, key);
120 }