Logo AND Algorithmique Numérique Distribuée

Public GIT Repository
[simix] Use std::string for s_smx_process_arg
[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, "A dict with keys of type (name, level)");
14
15 xbt_lib_t xbt_lib_new(void)
16 {
17   xbt_lib_t lib  = xbt_new(s_xbt_lib_t, 1);
18   lib->dict = xbt_dict_new_homogeneous(xbt_free_f);
19   lib->levels = 0;
20   lib->free_f = NULL;
21   return lib;
22 }
23
24 void xbt_lib_free(xbt_lib_t *plib)
25 {
26   xbt_lib_t lib = *plib;
27   if (lib) {
28     xbt_dict_cursor_t cursor;
29     char *key;
30     void **elts;
31     xbt_dict_foreach (lib->dict, cursor, key, elts) {
32       int i;
33       for (i = 0 ; i < lib->levels ; i++)
34         if (elts[i] && lib->free_f[i])
35           lib->free_f[i](elts[i]);
36     }
37     xbt_dict_free(&lib->dict);
38     xbt_free(lib->free_f);
39     xbt_free(lib);
40     *plib = NULL;
41   }
42 }
43
44 int xbt_lib_add_level(xbt_lib_t lib, void_f_pvoid_t free_f)
45 {
46   XBT_DEBUG("xbt_lib_add_level");
47   xbt_assert(xbt_dict_is_empty(lib->dict), "Lib is not empty, cannot add a level");
48   lib->free_f = xbt_realloc(lib->free_f, sizeof(void_f_pvoid_t) * (lib->levels + 1));
49   lib->free_f[lib->levels] = free_f;
50   return lib->levels++;
51 }
52
53 void xbt_lib_set(xbt_lib_t lib, const char *key, int level, void *obj)
54 {
55   XBT_DEBUG("xbt_lib_set key '%s:%d' with object %p", key, level, obj);
56   void **elts = xbt_dict_get_or_null(lib->dict, key);
57   if (!elts) {
58     elts = xbt_new0(void *, lib->levels);
59     xbt_dict_set(lib->dict, key, elts, NULL);
60   }
61   if (elts[level]) {
62     XBT_DEBUG("Replace %p by %p element under key '%s:%d'", elts[level], obj, key, level);
63     if (lib->free_f[level])
64       lib->free_f[level](elts[level]);
65   }
66   elts[level] = obj;
67 }
68
69 void xbt_lib_unset(xbt_lib_t lib, const char *key, int level, int invoke_callback)
70 {
71   void **elts = xbt_dict_get_or_null(lib->dict, key);
72   if (!elts) {
73      XBT_WARN("no key %s", key);
74      return;
75   }
76
77   void *obj = elts[level];
78   if (!obj) {
79      XBT_WARN("no key %s at level %d", key, level);
80      return;
81   }
82
83   XBT_DEBUG("Remove %p of key %s at level %d", obj, key, level);
84   elts[level] = NULL;
85
86   /* check if there still remains any elements of this key */
87   int empty = 1;
88   for (int i = 0; i < lib->levels && empty; i++) {
89      if (elts[i] != NULL)
90        empty = 0;
91   }
92   if (empty) {
93     /* there is no element at any level, so delete the key */
94     xbt_dict_remove(lib->dict, key);
95   }
96
97   if (invoke_callback && lib->free_f[level])
98     lib->free_f[level](obj);
99 }
100
101 void *xbt_lib_get_or_null(xbt_lib_t lib, const char *key, int level)
102 {
103   void **elts = xbt_dict_get_or_null(lib->dict, key);
104   return elts ? elts[level] : NULL;
105 }
106
107 xbt_dictelm_t xbt_lib_get_elm_or_null(xbt_lib_t lib, const char *key)
108 {
109   return xbt_dict_get_elm_or_null(lib->dict, key);
110 }
111
112 void *xbt_lib_get_level(xbt_dictelm_t elm, int level){
113   void **elts = elm->content;
114   return elts ? elts[level] : NULL;
115 }
116
117 void xbt_lib_remove(xbt_lib_t lib, const char *key){
118   xbt_dict_remove(lib->dict, key);
119 }