| #ifndef STRMAP_H |
| #define STRMAP_H |
| |
| #include "hashmap.h" |
| |
| struct mem_pool; |
| struct strmap { |
| struct hashmap map; |
| struct mem_pool *pool; |
| unsigned int strdup_strings:1; |
| }; |
| |
| struct strmap_entry { |
| struct hashmap_entry ent; |
| const char *key; |
| void *value; |
| /* strmap_entry may be allocated extra space to store the key at end */ |
| }; |
| |
| int cmp_strmap_entry(const void *hashmap_cmp_fn_data, |
| const struct hashmap_entry *entry1, |
| const struct hashmap_entry *entry2, |
| const void *keydata); |
| |
| #define STRMAP_INIT { \ |
| .map = HASHMAP_INIT(cmp_strmap_entry, NULL), \ |
| .strdup_strings = 1, \ |
| } |
| #define STRINTMAP_INIT { \ |
| .map = STRMAP_INIT, \ |
| .default_value = 0, \ |
| } |
| #define STRSET_INIT { .map = STRMAP_INIT } |
| |
| /* |
| * Initialize the members of the strmap. Any keys added to the strmap will |
| * be strdup'ed with their memory managed by the strmap. |
| */ |
| void strmap_init(struct strmap *map); |
| |
| /* |
| * Same as strmap_init, but for those who want to control the memory management |
| * carefully instead of using the default of strdup_strings=1 and pool=NULL. |
| */ |
| void strmap_init_with_options(struct strmap *map, |
| struct mem_pool *pool, |
| int strdup_strings); |
| |
| /* |
| * Remove all entries from the map, releasing any allocated resources. |
| */ |
| void strmap_clear(struct strmap *map, int free_values); |
| |
| /* |
| * Similar to strmap_clear() but leaves map->map->table allocated and |
| * pre-sized so that subsequent uses won't need as many rehashings. |
| */ |
| void strmap_partial_clear(struct strmap *map, int free_values); |
| |
| /* |
| * Insert "str" into the map, pointing to "data". |
| * |
| * If an entry for "str" already exists, its data pointer is overwritten, and |
| * the original data pointer returned. Otherwise, returns NULL. |
| */ |
| void *strmap_put(struct strmap *map, const char *str, void *data); |
| |
| /* |
| * Return the strmap_entry mapped by "str", or NULL if there is not such |
| * an item in map. |
| */ |
| struct strmap_entry *strmap_get_entry(struct strmap *map, const char *str); |
| |
| /* |
| * Return the data pointer mapped by "str", or NULL if the entry does not |
| * exist. |
| */ |
| void *strmap_get(struct strmap *map, const char *str); |
| |
| /* |
| * Return non-zero iff "str" is present in the map. This differs from |
| * strmap_get() in that it can distinguish entries with a NULL data pointer. |
| */ |
| int strmap_contains(struct strmap *map, const char *str); |
| |
| /* |
| * Remove the given entry from the strmap. If the string isn't in the |
| * strmap, the map is not altered. |
| */ |
| void strmap_remove(struct strmap *map, const char *str, int free_value); |
| |
| /* |
| * Return how many entries the strmap has. |
| */ |
| static inline unsigned int strmap_get_size(struct strmap *map) |
| { |
| return hashmap_get_size(&map->map); |
| } |
| |
| /* |
| * Return whether the strmap is empty. |
| */ |
| static inline int strmap_empty(struct strmap *map) |
| { |
| return strmap_get_size(map) == 0; |
| } |
| |
| /* |
| * iterate through @map using @iter, @var is a pointer to a type strmap_entry |
| */ |
| #define strmap_for_each_entry(mystrmap, iter, var) \ |
| hashmap_for_each_entry(&(mystrmap)->map, iter, var, ent) |
| |
| |
| /* |
| * strintmap: |
| * A map of string -> int, typecasting the void* of strmap to an int. |
| * |
| * Primary differences: |
| * 1) Since the void* value is just an int in disguise, there is no value |
| * to free. (Thus one fewer argument to strintmap_clear) |
| * 2) strintmap_get() returns an int, or returns the default_value if the |
| * key is not found in the strintmap. |
| * 3) No strmap_put() equivalent; strintmap_set() and strintmap_incr() |
| * instead. |
| */ |
| |
| struct strintmap { |
| struct strmap map; |
| int default_value; |
| }; |
| |
| #define strintmap_for_each_entry(mystrmap, iter, var) \ |
| strmap_for_each_entry(&(mystrmap)->map, iter, var) |
| |
| static inline void strintmap_init(struct strintmap *map, int default_value) |
| { |
| strmap_init(&map->map); |
| map->default_value = default_value; |
| } |
| |
| static inline void strintmap_init_with_options(struct strintmap *map, |
| int default_value, |
| struct mem_pool *pool, |
| int strdup_strings) |
| { |
| strmap_init_with_options(&map->map, pool, strdup_strings); |
| map->default_value = default_value; |
| } |
| |
| static inline void strintmap_clear(struct strintmap *map) |
| { |
| strmap_clear(&map->map, 0); |
| } |
| |
| static inline void strintmap_partial_clear(struct strintmap *map) |
| { |
| strmap_partial_clear(&map->map, 0); |
| } |
| |
| static inline int strintmap_contains(struct strintmap *map, const char *str) |
| { |
| return strmap_contains(&map->map, str); |
| } |
| |
| static inline void strintmap_remove(struct strintmap *map, const char *str) |
| { |
| return strmap_remove(&map->map, str, 0); |
| } |
| |
| static inline int strintmap_empty(struct strintmap *map) |
| { |
| return strmap_empty(&map->map); |
| } |
| |
| static inline unsigned int strintmap_get_size(struct strintmap *map) |
| { |
| return strmap_get_size(&map->map); |
| } |
| |
| /* |
| * Returns the value for str in the map. If str isn't found in the map, |
| * the map's default_value is returned. |
| */ |
| static inline int strintmap_get(struct strintmap *map, const char *str) |
| { |
| struct strmap_entry *result = strmap_get_entry(&map->map, str); |
| if (!result) |
| return map->default_value; |
| return (intptr_t)result->value; |
| } |
| |
| static inline void strintmap_set(struct strintmap *map, const char *str, |
| intptr_t v) |
| { |
| strmap_put(&map->map, str, (void *)v); |
| } |
| |
| /* |
| * Increment the value for str by amt. If str isn't in the map, add it and |
| * set its value to default_value + amt. |
| */ |
| void strintmap_incr(struct strintmap *map, const char *str, intptr_t amt); |
| |
| /* |
| * strset: |
| * A set of strings. |
| * |
| * Primary differences with strmap: |
| * 1) The value is always NULL, and ignored. As there is no value to free, |
| * there is one fewer argument to strset_clear |
| * 2) No strset_get() because there is no value. |
| * 3) No strset_put(); use strset_add() instead. |
| */ |
| |
| struct strset { |
| struct strmap map; |
| }; |
| |
| #define strset_for_each_entry(mystrset, iter, var) \ |
| strmap_for_each_entry(&(mystrset)->map, iter, var) |
| |
| static inline void strset_init(struct strset *set) |
| { |
| strmap_init(&set->map); |
| } |
| |
| static inline void strset_init_with_options(struct strset *set, |
| struct mem_pool *pool, |
| int strdup_strings) |
| { |
| strmap_init_with_options(&set->map, pool, strdup_strings); |
| } |
| |
| static inline void strset_clear(struct strset *set) |
| { |
| strmap_clear(&set->map, 0); |
| } |
| |
| static inline void strset_partial_clear(struct strset *set) |
| { |
| strmap_partial_clear(&set->map, 0); |
| } |
| |
| static inline int strset_contains(struct strset *set, const char *str) |
| { |
| return strmap_contains(&set->map, str); |
| } |
| |
| static inline void strset_remove(struct strset *set, const char *str) |
| { |
| return strmap_remove(&set->map, str, 0); |
| } |
| |
| static inline int strset_empty(struct strset *set) |
| { |
| return strmap_empty(&set->map); |
| } |
| |
| static inline unsigned int strset_get_size(struct strset *set) |
| { |
| return strmap_get_size(&set->map); |
| } |
| |
| /* Returns 1 if str is added to the set; returns 0 if str was already in set */ |
| int strset_add(struct strset *set, const char *str); |
| |
| #endif /* STRMAP_H */ |