C Container Collection (CCC)
Loading...
Searching...
No Matches
flat_hash_map.h File Reference

The Flat Hash Map Interface. More...

#include "impl/impl_flat_hash_map.h"
#include "types.h"
Include dependency graph for flat_hash_map.h:
This graph shows which files directly or indirectly include this file:

Go to the source code of this file.

Detailed Description

The Flat Hash Map Interface.

A Flat Hash Map stores elements by hash value and allows the user to retrieve them by key in amortized O(1). Elements in the table may be copied and moved, especially when rehashing occurs, so no pointer stability is available in this implementation.

A flat hash map requires the user to provide a pointer to the map, a type, a key field, a hash function, and a key comparator function. The hash function should be well tailored to the key being stored in the table to prevent collisions. Good variety in the upper bits of the hashed value will also result in faster performance. Currently, the flat hash map does not offer any default hash functions or hash strengthening algorithms so strong hash functions should be obtained by the user for the data set.

To shorten names in the interface, define the following preprocessor directive at the top of your file.

#define FLAT_HASH_MAP_USING_NAMESPACE_CCC

All types and functions can then be written without the ccc_ prefix.

Initialization Interface

Initialize the container with memory, callbacks, and permissions. When a fixed size map is required that will not have allocation permission, the user must declare the type name and size of the map they will use.

#define ccc_fhm_declare_fixed_map(fixed_map_type_name, key_val_type_name, capacity)
 Declare a fixed size map type for use in the stack, heap, or data segment. Does not return a value.
 
#define ccc_fhm_fixed_capacity(fixed_map_type_name)    ccc_impl_fhm_fixed_capacity(fixed_map_type_name)
 Obtain the capacity previously chosen for the fixed size map type.
 
#define ccc_fhm_init(map_ptr, any_type_name, key_field, hash_fn, key_eq_fn, alloc_fn, aux_data, capacity)
 Initialize a map with a buffer of types at compile time or runtime.
 
ccc_result ccc_fhm_copy (ccc_flat_hash_map *dst, ccc_flat_hash_map const *src, ccc_any_alloc_fn *fn)
 Copy the map at source to destination.
 
ccc_result ccc_fhm_reserve (ccc_flat_hash_map *h, size_t to_add, ccc_any_alloc_fn *fn)
 Reserve space required to add a specified number of elements to the map. If the current capacity is sufficient, do nothing.
 

Entry Interface

Obtain and operate on container entries for efficient queries when non-trivial control flow is needed.

#define ccc_fhm_entry_r(map_ptr, key_ptr)
 Obtains an entry for the provided key in the table for future use.
 
#define ccc_fhm_and_modify_w(map_entry_ptr, type_name, closure_over_T...)
 Modify an Occupied entry with a closure over user type T.
 
#define ccc_fhm_or_insert_w(map_entry_ptr, lazy_key_value...)    ccc_impl_fhm_or_insert_w(map_entry_ptr, lazy_key_value)
 lazily insert the desired key value into the entry if it is Vacant.
 
#define ccc_fhm_insert_entry_w(map_entry_ptr, lazy_key_value...)    ccc_impl_fhm_insert_entry_w(map_entry_ptr, lazy_key_value)
 write the contents of the compound literal lazy_key_value to a slot.
 
#define ccc_fhm_swap_entry_r(map_ptr, key_val_type_ptr)
 Invariantly inserts the key value wrapping out_handle.
 
#define ccc_fhm_remove_entry_r(map_entry_ptr)
 Remove the entry from the table if Occupied.
 
#define ccc_fhm_try_insert_r(map_ptr, key_val_type_ptr)
 Attempts to insert the key value wrapping key_val_handle_ptr.
 
#define ccc_fhm_try_insert_w(map_ptr, key, lazy_value...)
 lazily insert lazy_value into the map at key if key is absent.
 
#define ccc_fhm_insert_or_assign_r(map_ptr, key_val_type_ptr)
 Invariantly inserts or overwrites a user struct into the table.
 
#define ccc_fhm_insert_or_assign_w(map_ptr, key, lazy_value...)
 Inserts a new key value pair or overwrites the existing entry.
 
#define ccc_fhm_remove_r(map_ptr, key_val_type_output_ptr)
 Removes the key value in the map storing the old value, if present, in the struct containing out_handle_ptr provided by the user.
 
ccc_fhmap_entry ccc_fhm_entry (ccc_flat_hash_map *h, void const *key)
 Obtains an entry for the provided key in the table for future use.
 
ccc_fhmap_entry * ccc_fhm_and_modify (ccc_fhmap_entry *e, ccc_any_type_update_fn *fn)
 Modifies the provided entry if it is Occupied.
 
ccc_fhmap_entry * ccc_fhm_and_modify_aux (ccc_fhmap_entry *e, ccc_any_type_update_fn *fn, void *aux)
 Modifies the provided entry if it is Occupied.
 
void * ccc_fhm_or_insert (ccc_fhmap_entry const *e, void const *key_val_type)
 Inserts the struct with handle elem if the entry is Vacant.
 
void * ccc_fhm_insert_entry (ccc_fhmap_entry const *e, void const *key_val_type)
 Inserts the provided entry invariantly.
 
ccc_entry ccc_fhm_swap_entry (ccc_flat_hash_map *h, void *key_val_type_output)
 Invariantly inserts the key value wrapping out_handle.
 
ccc_entry ccc_fhm_remove_entry (ccc_fhmap_entry const *e)
 Remove the entry from the table if Occupied.
 
ccc_entry ccc_fhm_try_insert (ccc_flat_hash_map *h, void *key_val_type)
 Attempts to insert the key value wrapping key_val_handle.
 
ccc_entry ccc_fhm_insert_or_assign (ccc_flat_hash_map *h, void *key_val_type)
 Invariantly inserts or overwrites a user struct into the table.
 
ccc_entry ccc_fhm_remove (ccc_flat_hash_map *h, void *key_val_type_output)
 Removes the key value in the map storing the old value, if present, in the struct containing out_handle provided by the user.
 
void * ccc_fhm_unwrap (ccc_fhmap_entry const *e)
 Unwraps the provided entry to obtain a view into the table element.
 
ccc_tribool ccc_fhm_occupied (ccc_fhmap_entry const *e)
 Returns the Vacant or Occupied status of the entry.
 
ccc_tribool ccc_fhm_insert_error (ccc_fhmap_entry const *e)
 Provides the status of the entry should an insertion follow.
 
ccc_entry_status ccc_fhm_entry_status (ccc_fhmap_entry const *e)
 Obtain the entry status from a container entry.
 

Container Types

Types available in the container interface.

typedef struct ccc_fhmap ccc_flat_hash_map
 A container for storing key-value structures defined by the user in a contiguous buffer.
 
typedef union ccc_fhmap_entry ccc_fhmap_entry
 A container specific entry used to implement the Entry Interface.
 

Membership Interface

Test membership or obtain references to stored user types directly.

ccc_tribool ccc_fhm_contains (ccc_flat_hash_map const *h, void const *key)
 Searches the table for the presence of key.
 
void * ccc_fhm_get_key_val (ccc_flat_hash_map const *h, void const *key)
 Returns a reference into the table at entry key.
 

Deallocation Interface

Destroy the container.

ccc_result ccc_fhm_clear (ccc_flat_hash_map *h, ccc_any_type_destructor_fn *fn)
 Frees all slots in the table for use without affecting capacity.
 
ccc_result ccc_fhm_clear_and_free (ccc_flat_hash_map *h, ccc_any_type_destructor_fn *fn)
 Frees all slots in the table and frees the underlying buffer.
 
ccc_result ccc_fhm_clear_and_free_reserve (ccc_flat_hash_map *h, ccc_any_type_destructor_fn *destructor, ccc_any_alloc_fn *alloc)
 Frees all slots in the table and frees the underlying buffer that was previously dynamically reserved with the reserve function.
 

Iterator Interface

Obtain and manage iterators over the container.

void * ccc_fhm_begin (ccc_flat_hash_map const *h)
 Obtains a pointer to the first element in the table.
 
void * ccc_fhm_next (ccc_flat_hash_map const *h, void const *key_val_type_iter)
 Advances the iterator to the next occupied table slot.
 
void * ccc_fhm_end (ccc_flat_hash_map const *h)
 Check the current iterator against the end for loop termination.
 

State Interface

Obtain the container state.

ccc_tribool ccc_fhm_is_empty (ccc_flat_hash_map const *h)
 Returns the size status of the table.
 
ccc_ucount ccc_fhm_size (ccc_flat_hash_map const *h)
 Returns the size of the table representing active slots.
 
ccc_ucount ccc_fhm_capacity (ccc_flat_hash_map const *h)
 Return the full capacity of the backing storage.
 
void * ccc_fhm_data (ccc_flat_hash_map const *h)
 Return a reference to the base of backing array. O(1).
 
ccc_tribool ccc_fhm_validate (ccc_flat_hash_map const *h)
 Validation of invariants for the hash table.
 

Macro Definition Documentation

◆ ccc_fhm_and_modify_w

#define ccc_fhm_and_modify_w (   map_entry_ptr,
  type_name,
  closure_over_T... 
)
Value:
{ \
ccc_impl_fhm_and_modify_w(map_entry_ptr, type_name, closure_over_T) \
}
union ccc_fhmap_entry ccc_fhmap_entry
A container specific entry used to implement the Entry Interface.
Definition: flat_hash_map.h:65

Modify an Occupied entry with a closure over user type T.

Parameters
[in]map_entry_ptra pointer to the obtained entry.
[in]type_namethe name of the user type stored in the container.
[in]closure_over_Tthe code to be run on the reference to user type T, if Occupied. This may be a semicolon separated list of statements to execute on T or a section of code wrapped in braces {code here} which may be preferred for formatting.
Returns
a compound literal reference to the modified entry if it was occupied or a vacant entry if it was vacant.
Note
T is a reference to the user type stored in the entry guaranteed to be non-NULL if the closure executes.
#define FLAT_HASH_MAP_USING_NAMESPACE_CCC
fhm_entry *e = fhm_and_modify_w(entry_r(&fhm, &k), word, T->cnt++;);
word *w = fhm_or_insert_w(fhm_and_modify_w(entry_r(&fhm, &k), word,
{ T->cnt++; }),
(word){.key = k, .cnt = 1});

Note that any code written is only evaluated if the entry is Occupied and the container can deliver the user type T. This means any function calls are lazily evaluated in the closure scope.

◆ ccc_fhm_declare_fixed_map

#define ccc_fhm_declare_fixed_map (   fixed_map_type_name,
  key_val_type_name,
  capacity 
)
Value:
ccc_impl_fhm_declare_fixed_map(fixed_map_type_name, key_val_type_name, \
capacity)

Declare a fixed size map type for use in the stack, heap, or data segment. Does not return a value.

Parameters
[in]fixed_map_type_namethe user chosen name of the fixed sized map.
[in]key_val_type_namethe type the user plans to store in the map. It may have a key and value field as well as any additional fields. For set-like behavior, wrap a field in a struct/union (e.g. union int_elem {int e;};).
[in]capacitythe power of two capacity for the map.
Warning
the capacity must be a power of two greater than 8 or 16, depending on the platform (e.g. 16, 32, 64, etc.).

Once the location for the fixed size map is chosen–stack, heap, or data segment–provide a pointer to the map for the initialization macro.

struct val
{
int key;
int val;
};
ccc_fhm_declare_fixed_map(small_fixed_map, struct val, 64);
static flat_hash_map static_fh = fhm_init(
&(static small_fixed_map){},
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
NULL,
NULL,
fhm_fixed_capacity(small_fixed_map)
);
#define ccc_fhm_declare_fixed_map(fixed_map_type_name, key_val_type_name, capacity)
Declare a fixed size map type for use in the stack, heap, or data segment. Does not return a value.
Definition: flat_hash_map.h:139

Similarly, a fixed size map can be used on the stack.

struct val
{
int key;
int val;
};
ccc_fhm_declare_fixed_map(small_fixed_map, struct val, 64);
int main(void)
{
flat_hash_map static_fh = fhm_init(
&(small_fixed_map){},
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
NULL,
NULL,
fhm_fixed_capacity(small_fixed_map)
);
return 0;
}

The ccc_fhm_fixed_capacity macro can be used to obtain the previously provided capacity when declaring the fixed map type. Finally, one could allocate a fixed size map on the heap; however, it is usually better to initialize a dynamic map and use the ccc_fhm_reserve function for such a use case.

This macro is not needed when a dynamic resizing flat hash map is needed. For dynamic maps, simply pass NULL and 0 capacity to the initialization macro.

◆ ccc_fhm_entry_r

#define ccc_fhm_entry_r (   map_ptr,
  key_ptr 
)
Value:
{ \
ccc_fhm_entry(map_ptr, key_ptr).impl \
}

Obtains an entry for the provided key in the table for future use.

Parameters
[in]map_ptrthe hash table to be searched.
[in]key_ptrthe key used to search the table matching the stored key type.
Returns
a compound literal reference to a specialized hash entry for use with other functions in the Entry Interface.
Warning
the contents of an entry should not be examined or modified. Use the provided functions, only.

An entry is a search result that provides either an Occupied or Vacant entry in the table. An occupied entry signifies that the search was successful. A Vacant entry means the search was not successful but we now have a handle to where in the table such an element should be inserted.

An entry is most often passed in a functional style to subsequent calls in the Entry Interface.

◆ ccc_fhm_fixed_capacity

#define ccc_fhm_fixed_capacity (   fixed_map_type_name)     ccc_impl_fhm_fixed_capacity(fixed_map_type_name)

Obtain the capacity previously chosen for the fixed size map type.

Parameters
[in]fixed_map_type_namethe name of a previously declared map.
Returns
the size_t capacity. This is the full capacity without any load factor restrictions.

◆ ccc_fhm_init

#define ccc_fhm_init (   map_ptr,
  any_type_name,
  key_field,
  hash_fn,
  key_eq_fn,
  alloc_fn,
  aux_data,
  capacity 
)
Value:
ccc_impl_fhm_init(map_ptr, any_type_name, key_field, hash_fn, key_eq_fn, \
alloc_fn, aux_data, capacity)

Initialize a map with a buffer of types at compile time or runtime.

Parameters
[in]map_ptra pointer to a fixed map allocation or NULL.
[in]any_type_namethe name of the user defined type stored in the map.
[in]key_fieldthe field of the struct used for key storage.
[in]hash_fnthe ccc_any_key_hash_fn function the user desires for the table.
[in]key_eq_fnthe ccc_any_key_eq_fn the user intends to use.
[in]alloc_fnthe allocation function for resizing or NULL if no resizing is allowed.
[in]aux_dataauxiliary data that is needed for hashing or comparison.
[in]capacitythe capacity of a fixed size map or 0.
Returns
the flat hash map directly initialized on the right hand side of the equality operator (i.e. ccc_flat_hash_map fh = ccc_fhm_init(...);)
Note
if a dynamic resizing map is required provide NULL as the map_ptr.

Initialize a static fixed size hash map at compile time that has no allocation permission or auxiliary data needed.

#define FLAT_HASH_MAP_USING_NAMESPACE_CCC
struct val
{
int key;
int val;
};
fhm_declare_fixed_map(small_fixed_map, struct val, 64);
static flat_hash_map static_fh = fhm_init(
&(static small_fixed_map){},
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
NULL,
NULL,
fhm_fixed_capacity(small_fixed_map)
);

Initialize a dynamic hash table at compile time with allocation permission and no auxiliary data. Use the same type as the previous example.

#define FLAT_HASH_MAP_USING_NAMESPACE_CCC
struct val
{
int key;
int val;
};
static flat_hash_map static_fh = fhm_init(
NULL,
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
std_alloc,
NULL,
0
);

Initialization at runtime is also possible. Stack-based or dynamic maps are identical to the provided examples. Omit static in a runtime context.

◆ ccc_fhm_insert_entry_w

#define ccc_fhm_insert_entry_w (   map_entry_ptr,
  lazy_key_value... 
)     ccc_impl_fhm_insert_entry_w(map_entry_ptr, lazy_key_value)

write the contents of the compound literal lazy_key_value to a slot.

Parameters
[in]map_entry_ptra pointer to the obtained entry.
[in]lazy_key_valuethe compound literal to write to a new slot.
Returns
a reference to the newly inserted or overwritten user type. NULL is returned if resizing is required but fails or is not allowed.

◆ ccc_fhm_insert_or_assign_r

#define ccc_fhm_insert_or_assign_r (   map_ptr,
  key_val_type_ptr 
)
Value:
&(ccc_entry) \
{ \
ccc_fhm_insert_or_assign(map_ptr, key_val_type_ptr).impl \
}
#define ccc_entry(container_ptr, key_ptr...)
Obtain a container specific entry for the Entry Interface.
Definition: traits.h:141

Invariantly inserts or overwrites a user struct into the table.

Parameters
[in]map_ptra pointer to the flat hash map.
[in]key_val_type_ptrthe complete key and value type to be inserted.
Returns
a compound literal reference to the entry. If Occupied an entry was overwritten by the new key value. If Vacant no prior table entry existed.

Note that this function can be used when the old user type is not needed but the information regarding its presence is helpful.

◆ ccc_fhm_insert_or_assign_w

#define ccc_fhm_insert_or_assign_w (   map_ptr,
  key,
  lazy_value... 
)
Value:
&(ccc_entry) \
{ \
ccc_impl_fhm_insert_or_assign_w(map_ptr, key, lazy_value) \
}

Inserts a new key value pair or overwrites the existing entry.

Parameters
[in]map_ptrthe pointer to the flat hash map.
[in]keythe key to be searched in the table.
[in]lazy_valuethe compound literal specifying the value.
Returns
a compound literal reference to the entry of the existing or newly inserted value. Occupied indicates the key existed, Vacant indicates the key was absent. Unwrapping in any case provides the current value unless an error occurs that prevents insertion. An insertion error will flag such a case.

Note that for brevity and convenience the user need not write the key to the lazy value compound literal as well. This function ensures the key in the compound literal matches the searched key.

◆ ccc_fhm_or_insert_w

#define ccc_fhm_or_insert_w (   map_entry_ptr,
  lazy_key_value... 
)     ccc_impl_fhm_or_insert_w(map_entry_ptr, lazy_key_value)

lazily insert the desired key value into the entry if it is Vacant.

Parameters
[in]map_entry_ptra pointer to the obtained entry.
[in]lazy_key_valuethe compound literal to construct in place if the entry is Vacant.
Returns
a reference to the unwrapped user type in the entry, either the unmodified reference if the entry was Occupied or the newly inserted element if the entry was Vacant. NULL is returned if resizing is required but fails or is not allowed.

Note that if the compound literal uses any function calls to generate values or other data, such functions will not be called if the entry is Occupied.

◆ ccc_fhm_remove_entry_r

#define ccc_fhm_remove_entry_r (   map_entry_ptr)
Value:
&(ccc_entry) \
{ \
ccc_fhm_remove_entry(map_entry_ptr).impl \
}

Remove the entry from the table if Occupied.

Parameters
[in]map_entry_ptra pointer to the table entry.
Returns
a compound liter to an entry containing NULL. If Occupied an entry in the table existed and was removed. If Vacant, no prior entry existed to be removed.

◆ ccc_fhm_remove_r

#define ccc_fhm_remove_r (   map_ptr,
  key_val_type_output_ptr 
)
Value:
&(ccc_entry) \
{ \
ccc_fhm_remove(map_ptr, key_val_type_output_ptr).impl \
}

Removes the key value in the map storing the old value, if present, in the struct containing out_handle_ptr provided by the user.

Parameters
[in]map_ptrthe pointer to the flat hash map.
[out]key_val_type_output_ptrthe complete key and value type to be removed
Returns
a compound literal reference to the removed entry. If Occupied it may be unwrapped to obtain the old key value pair. If Vacant the key value pair was not stored in the map. If bad input is provided an input error is set. If a previously stored value is returned it is safe to keep and modify this reference because the data has been written to the provided space.

Note that this function may write to the struct containing the second parameter and wraps it in an entry to provide information about the old value.

◆ ccc_fhm_swap_entry_r

#define ccc_fhm_swap_entry_r (   map_ptr,
  key_val_type_ptr 
)
Value:
&(ccc_entry) \
{ \
ccc_fhm_swap_entry(map_ptr, key_val_type_ptr).impl \
}

Invariantly inserts the key value wrapping out_handle.

Parameters
[in]map_ptrthe pointer to the flat hash map.
[out]key_val_type_ptrthe complete key and value type to be inserted.
Returns
a compound literal reference to an entry. If Vacant, no prior element with key existed and entry may be unwrapped to view the new insertion in the table. If Occupied the old value is written to key_val_type_output and may be unwrapped to view. If more space is needed but allocation fails or has been forbidden, an insert error is set.

Note that this function may write to the struct containing the second parameter and wraps it in an entry to provide information about the old value.

◆ ccc_fhm_try_insert_r

#define ccc_fhm_try_insert_r (   map_ptr,
  key_val_type_ptr 
)
Value:
&(ccc_entry) \
{ \
ccc_fhm_try_insert(map_ptr, key_val_type_ptr).impl \
}

Attempts to insert the key value wrapping key_val_handle_ptr.

Parameters
[in]map_ptrthe pointer to the flat hash map.
[in]key_val_type_ptrthe complete key and value type to be inserted.
Returns
a compound literal reference to the entry. If Occupied, the entry contains a reference to the key value user type in the table and may be unwrapped. If Vacant the entry contains a reference to the newly inserted entry in the table. If more space is needed but allocation fails or has been forbidden, an insert error is set.
Warning
because this function returns a reference to a user type in the table any subsequent insertions or deletions invalidate this reference.

◆ ccc_fhm_try_insert_w

#define ccc_fhm_try_insert_w (   map_ptr,
  key,
  lazy_value... 
)
Value:
&(ccc_entry) \
{ \
ccc_impl_fhm_try_insert_w(map_ptr, key, lazy_value) \
}

lazily insert lazy_value into the map at key if key is absent.

Parameters
[in]map_ptra pointer to the flat hash map.
[in]keythe direct key r-value.
[in]lazy_valuethe compound literal specifying the value.
Returns
a compound literal reference to the entry of the existing or newly inserted value. Occupied indicates the key existed, Vacant indicates the key was absent. Unwrapping in any case provides the current value unless an error occurs that prevents insertion. An insertion error will flag such a case.
Warning
ensure the key type matches the type stored in table as your key. For example, if your key is of type int and you pass a size_t variable to the key argument, you will overwrite adjacent bytes of your struct.

Note that for brevity and convenience the user need not write the key to the lazy value compound literal as well. This function ensures the key in the compound literal matches the searched key.

Typedef Documentation

◆ ccc_fhmap_entry

typedef union ccc_fhmap_entry ccc_fhmap_entry

A container specific entry used to implement the Entry Interface.

The Entry Interface offers efficient search and subsequent insertion, deletion, or value update based on the needs of the user.

◆ ccc_flat_hash_map

typedef struct ccc_fhmap ccc_flat_hash_map

A container for storing key-value structures defined by the user in a contiguous buffer.

A flat hash map can be initialized on the stack, heap, or data segment at runtime or compile time.

Function Documentation

◆ ccc_fhm_and_modify()

ccc_fhmap_entry * ccc_fhm_and_modify ( ccc_fhmap_entry *  e,
ccc_any_type_update_fn fn 
)

Modifies the provided entry if it is Occupied.

Parameters
[in]ethe entry obtained from an entry function or macro.
[in]fnan update function in which the auxiliary argument is unused.
Returns
the updated entry if it was Occupied or the unmodified vacant entry.

This function is intended to make the function chaining in the Entry Interface more succinct if the entry will be modified in place based on its own value without the need of the auxiliary argument a ccc_any_type_update_fn can provide.

◆ ccc_fhm_and_modify_aux()

ccc_fhmap_entry * ccc_fhm_and_modify_aux ( ccc_fhmap_entry *  e,
ccc_any_type_update_fn fn,
void *  aux 
)

Modifies the provided entry if it is Occupied.

Parameters
[in]ethe entry obtained from an entry function or macro.
[in]fnan update function that requires auxiliary data.
[in]auxauxiliary data required for the update.
Returns
the updated entry if it was Occupied or the unmodified vacant entry.

This function makes full use of a ccc_any_type_update_fn capability, meaning a complete ccc_update object will be passed to the update function callback.

◆ ccc_fhm_begin()

void * ccc_fhm_begin ( ccc_flat_hash_map const *  h)

Obtains a pointer to the first element in the table.

Parameters
[in]hthe table to iterate through.
Returns
a pointer that can be cast directly to the user type that is stored.
Warning
erasing or inserting during iteration may invalidate iterators if resizing occurs which would lead to undefined behavior. O(capacity).

Iteration starts from index 0 by capacity of the table so iteration order is not obvious to the user, nor should any specific order be relied on.

◆ ccc_fhm_capacity()

ccc_ucount ccc_fhm_capacity ( ccc_flat_hash_map const *  h)

Return the full capacity of the backing storage.

Parameters
[in]hthe hash table.
Returns
the capacity of the map or an argument error is set if h is NULL.

◆ ccc_fhm_clear()

ccc_result ccc_fhm_clear ( ccc_flat_hash_map h,
ccc_any_type_destructor_fn fn 
)

Frees all slots in the table for use without affecting capacity.

Parameters
[in]hthe table to be cleared.
[in]fnthe destructor for each element. NULL can be passed if no maintenance is required on the elements in the table before their slots are forfeit.

If NULL is passed as the destructor function time is O(1), else O(capacity).

◆ ccc_fhm_clear_and_free()

ccc_result ccc_fhm_clear_and_free ( ccc_flat_hash_map h,
ccc_any_type_destructor_fn fn 
)

Frees all slots in the table and frees the underlying buffer.

Parameters
[in]hthe table to be cleared.
[in]fnthe destructor for each element. NULL can be passed if no maintenance is required on the elements in the table before their slots are forfeit.
Returns
the result of free operation. If no alloc function is provided it is an error to attempt to free the buffer and a memory error is returned. Otherwise, an OK result is returned.

◆ ccc_fhm_clear_and_free_reserve()

ccc_result ccc_fhm_clear_and_free_reserve ( ccc_flat_hash_map h,
ccc_any_type_destructor_fn destructor,
ccc_any_alloc_fn alloc 
)

Frees all slots in the table and frees the underlying buffer that was previously dynamically reserved with the reserve function.

Parameters
[in]hthe table to be cleared.
[in]destructorthe destructor for each element. NULL can be passed if no maintenance is required on the elements in the table before their slots are forfeit.
[in]allocthe required allocation function to provide to a dynamically reserved map. Any auxiliary data provided upon initialization will be passed to the allocation function when called.
Returns
the result of free operation. CCC_RESULT_OK if success, or an error status to indicate the error.
Warning
It is an error to call this function on a map that was not reserved with the provided ccc_any_alloc_fn. The map must have existing memory to free.

This function covers the edge case of reserving a dynamic capacity for a map at runtime but denying the map allocation permission to resize. This can help prevent a map from growing unbounded due to internal decisions about rehashes and resizing. The user in this case knows the map does not have allocation permission and therefore no further memory will be dedicated to the map.

However, to free the map in such a case this function must be used because the map has no ability to free itself. Just as the allocation function is required to reserve memory so to is it required to free memory.

This function will work normally if called on a map with allocation permission however the normal ccc_fhm_clear_and_free is sufficient for that use case.

◆ ccc_fhm_contains()

ccc_tribool ccc_fhm_contains ( ccc_flat_hash_map const *  h,
void const *  key 
)

Searches the table for the presence of key.

Parameters
[in]hthe flat hash table to be searched.
[in]keypointer to the key matching the key type of the user struct.
Returns
true if the struct containing key is stored, false if not. Error if h or key is NULL.

◆ ccc_fhm_copy()

ccc_result ccc_fhm_copy ( ccc_flat_hash_map dst,
ccc_flat_hash_map const *  src,
ccc_any_alloc_fn fn 
)

Copy the map at source to destination.

Parameters
[in]dstthe initialized destination for the copy of the src map.
[in]srcthe initialized source of the map.
[in]fnthe optional allocation function if resizing is needed.
Returns
the result of the copy operation. If the destination capacity is less than the source capacity and no allocation function is provided an input error is returned. If resizing is required and resizing of dst fails a memory error is returned.
Note
dst must have capacity greater than or equal to src. If dst capacity is less than src, an allocation function must be provided with the fn argument.

Note that there are two ways to copy data from source to destination: provide sufficient memory and pass NULL as fn, or allow the copy function to take care of allocation for the copy.

Manual memory management with no allocation function provided.

#define FLAT_HASH_MAP_USING_NAMESPACE_CCC
struct val
{
int key;
int val;
};
fhm_declare_fixed_map(small_fixed_map, struct val, 64);
flat_hash_map src = fhm_init(
&(static small_fixed_map){},
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
NULL,
NULL,
ccc_fhm_fixed_capacity(small_fixed_map)
);
insert_rand_vals(&src);
flat_hash_map dst = fhm_init(
&(static small_fixed_map){},
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
NULL,
NULL,
ccc_fhm_fixed_capacity(small_fixed_map)
);
ccc_result res = fhm_copy(&dst, &src, NULL);
#define ccc_fhm_fixed_capacity(fixed_map_type_name)
Obtain the capacity previously chosen for the fixed size map type.
Definition: flat_hash_map.h:148
ccc_result
A result of actions on containers.
Definition: types.h:132

The above requires dst capacity be greater than or equal to src capacity. Here is memory management handed over to the copy function.

#define FLAT_HASH_MAP_USING_NAMESPACE_CCC
struct val
{
int key;
int val;
};
flat_hash_map src = fhm_init(
NULL,
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
std_alloc,
NULL,
0
);
insert_rand_vals(&src);
flat_hash_map dst = fhm_init(
NULL,
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
std_alloc,
NULL,
0
);
ccc_result res = fhm_copy(&dst, &src, std_alloc);

The above allows dst to have a capacity less than that of the src as long as copy has been provided an allocation function to resize dst. Note that this would still work if copying to a destination that the user wants as a fixed size map.

#define FLAT_HASH_MAP_USING_NAMESPACE_CCC
struct val
{
int key;
int val;
};
flat_hash_map src = fhm_init(
NULL,
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
std_alloc,
NULL,
0
);
insert_rand_vals(&src);
flat_hash_map dst = fhm_init(
NULL,
struct val,
key,
fhmap_int_to_u64,
fhmap_id_eq,
NULL,
NULL,
0
);
ccc_result res = fhm_copy(&dst, &src, std_alloc);

The above sets up dst with fixed size while src is a dynamic map. Because an allocation function is provided, the dst is resized once for the copy and retains its fixed size after the copy is complete. This would require the user to manually free the underlying buffer at dst eventually if this method is used. Usually it is better to allocate the memory with the reserve function if fixed size dynamic maps are required.

These options allow users to stay consistent across containers with their memory management strategies.

◆ ccc_fhm_data()

void * ccc_fhm_data ( ccc_flat_hash_map const *  h)

Return a reference to the base of backing array. O(1).

Parameters
[in]ha pointer to the map.
Returns
a reference to the base of the backing array.
Note
the reference is to the base of the backing array at index 0 with no consideration for the organization of map.
Warning
it is the users responsibility to ensure that access to any data is within the capacity of the backing buffer.

◆ ccc_fhm_end()

void * ccc_fhm_end ( ccc_flat_hash_map const *  h)

Check the current iterator against the end for loop termination.

Parameters
[in]hthe table being iterated upon.
Returns
the end address of the hash table.
Warning
It is undefined behavior to access or modify the end address.

◆ ccc_fhm_entry()

ccc_fhmap_entry ccc_fhm_entry ( ccc_flat_hash_map h,
void const *  key 
)

Obtains an entry for the provided key in the table for future use.

Parameters
[in]hthe hash table to be searched.
[in]keythe key used to search the table matching the stored key type.
Returns
a specialized hash entry for use with other functions in the Entry Interface.
Warning
the contents of an entry should not be examined or modified. Use the provided functions, only.

An entry is a search result that provides either an Occupied or Vacant entry in the table. An occupied entry signifies that the search was successful. A Vacant entry means the search was not successful but we now have a handle to where in the table such an element should be inserted.

An entry is rarely useful on its own. It should be passed in a functional style to subsequent calls in the Entry Interface.

◆ ccc_fhm_entry_status()

ccc_entry_status ccc_fhm_entry_status ( ccc_fhmap_entry const *  e)

Obtain the entry status from a container entry.

Parameters
[in]ea pointer to the entry.
Returns
the status stored in the entry after the required action on the container completes. If e is NULL an entry input error is returned so ensure e is non-NULL to avoid an inaccurate status returned.

Note that this function can be useful for debugging or if more detailed messages are needed for logging purposes. See ccc_entry_status_msg() in ccc/types.h for more information on detailed entry statuses.

◆ ccc_fhm_get_key_val()

void * ccc_fhm_get_key_val ( ccc_flat_hash_map const *  h,
void const *  key 
)

Returns a reference into the table at entry key.

Parameters
[in]hthe flat hash map to search.
[in]keythe key to search matching stored key type.
Returns
a view of the table entry if it is present, else NULL.

◆ ccc_fhm_insert_entry()

void * ccc_fhm_insert_entry ( ccc_fhmap_entry const *  e,
void const *  key_val_type 
)

Inserts the provided entry invariantly.

Parameters
[in]ethe entry returned from a call obtaining an entry.
[in]key_val_typethe complete key and value type to be inserted.
Returns
a pointer to the inserted element or NULL upon a memory error in which the load factor would be exceeded when no allocation policy is defined or resizing failed to find more memory.

This method can be used when the old value in the table does not need to be preserved. See the regular insert method if the old value is of interest. If an error occurs during the insertion process due to memory limitations or a search error NULL is returned. Otherwise insertion should not fail.

◆ ccc_fhm_insert_error()

ccc_tribool ccc_fhm_insert_error ( ccc_fhmap_entry const *  e)

Provides the status of the entry should an insertion follow.

Parameters
[in]ethe entry from a query to the table via function or macro.
Returns
true if the next insertion of a new element will cause an error. Error if e is null.

Table resizing occurs upon calls to entry functions/macros or when trying to insert a new element directly. This is to provide stable entries from the time they are obtained to the time they are used in functions they are passed to (i.e. the idiomatic or_insert(entry(...)...)).

However, if a Vacant entry is returned and then a subsequent insertion function is used, it will not work if resizing has failed and the return of those functions will indicate such a failure. One can also confirm an insertion error will occur from an entry with this function. For example, leaving this function in an assert for debug builds can be a helpful sanity check.

◆ ccc_fhm_insert_or_assign()

ccc_entry ccc_fhm_insert_or_assign ( ccc_flat_hash_map h,
void *  key_val_type 
)

Invariantly inserts or overwrites a user struct into the table.

Parameters
[in]ha pointer to the flat hash map.
[in]key_val_typethe complete key and value type to be inserted.
Returns
an entry. If Occupied an entry was overwritten by the new key value. If Vacant no prior table entry existed.

Note that this function can be used when the old user type is not needed but the information regarding its presence is helpful.

◆ ccc_fhm_is_empty()

ccc_tribool ccc_fhm_is_empty ( ccc_flat_hash_map const *  h)

Returns the size status of the table.

Parameters
[in]hthe hash table.
Returns
true if empty else false. Error if h is NULL.

◆ ccc_fhm_next()

void * ccc_fhm_next ( ccc_flat_hash_map const *  h,
void const *  key_val_type_iter 
)

Advances the iterator to the next occupied table slot.

Parameters
[in]hthe table being iterated upon.
[in]key_val_type_iterthe previous iterator.
Returns
a pointer that can be cast directly to the user type that is stored.
Warning
erasing or inserting during iteration may invalidate iterators if resizing occurs which would lead to undefined behavior. O(capacity).

◆ ccc_fhm_occupied()

ccc_tribool ccc_fhm_occupied ( ccc_fhmap_entry const *  e)

Returns the Vacant or Occupied status of the entry.

Parameters
[in]ethe entry from a query to the table via function or macro.
Returns
true if the entry is occupied, false if not. Error if e is NULL.

◆ ccc_fhm_or_insert()

void * ccc_fhm_or_insert ( ccc_fhmap_entry const *  e,
void const *  key_val_type 
)

Inserts the struct with handle elem if the entry is Vacant.

Parameters
[in]ethe entry obtained via function or macro call.
[in]key_val_typethe complete key and value type to be inserted.
Returns
a pointer to entry in the table invariantly. NULL on error.

Because this functions takes an entry and inserts if it is Vacant, the only reason NULL shall be returned is when an insertion error will occur, usually due to a resizing memory error. This can happen if the table is not allowed to resize because no allocation function is provided.

◆ ccc_fhm_remove()

ccc_entry ccc_fhm_remove ( ccc_flat_hash_map h,
void *  key_val_type_output 
)

Removes the key value in the map storing the old value, if present, in the struct containing out_handle provided by the user.

Parameters
[in]hthe pointer to the flat hash map.
[out]key_val_type_outputthe complete key and value type to be removed
Returns
the removed entry. If Occupied it may be unwrapped to obtain the old key value pair. If Vacant the key value pair was not stored in the map. If bad input is provided an input error is set. If a previously stored value is returned it is safe to keep and modify this reference because the data has been written to the provided space.

Note that this function may write to the struct containing the second parameter and wraps it in an entry to provide information about the old value.

◆ ccc_fhm_remove_entry()

ccc_entry ccc_fhm_remove_entry ( ccc_fhmap_entry const *  e)

Remove the entry from the table if Occupied.

Parameters
[in]ea pointer to the table entry.
Returns
an entry containing NULL. If Occupied an entry in the table existed and was removed. If Vacant, no prior entry existed to be removed.

◆ ccc_fhm_reserve()

ccc_result ccc_fhm_reserve ( ccc_flat_hash_map h,
size_t  to_add,
ccc_any_alloc_fn fn 
)

Reserve space required to add a specified number of elements to the map. If the current capacity is sufficient, do nothing.

Parameters
[in]ha pointer to the hash map.
[in]to_addthe number of elements to add to the map.
[in]fnthe required allocation function that can be used for resizing. Such a function is provided to cover the case where the user wants a fixed size map but cannot know the size needed until runtime. In this case, the function needs to be provided to allow the single resizing to occur. Any auxiliary data provided upon initialization will be passed to the allocation function when called.
Returns
the result of the reserving operation, OK if successful or an error code to indicate the specific failure.

If the map has already been initialized with allocation permission simply provide the same function that was passed upon initialization.

◆ ccc_fhm_size()

ccc_ucount ccc_fhm_size ( ccc_flat_hash_map const *  h)

Returns the size of the table representing active slots.

Parameters
[in]hthe hash table.
Returns
the size of the map or an argument error is set if h is NULL.

◆ ccc_fhm_swap_entry()

ccc_entry ccc_fhm_swap_entry ( ccc_flat_hash_map h,
void *  key_val_type_output 
)

Invariantly inserts the key value wrapping out_handle.

Parameters
[in]hthe pointer to the flat hash map.
[out]key_val_type_outputthe complete key and value type to be inserted.
Returns
an entry. If Vacant, no prior element with key existed and entry may be unwrapped to view the new insertion in the table. If Occupied the old value is written to key_val_type_output and may be unwrapped to view. If more space is needed but allocation fails or has been forbidden, an insert error is set.

Note that this function may write to the struct containing the second parameter and wraps it in an entry to provide information about the old value.

◆ ccc_fhm_try_insert()

ccc_entry ccc_fhm_try_insert ( ccc_flat_hash_map h,
void *  key_val_type 
)

Attempts to insert the key value wrapping key_val_handle.

Parameters
[in]hthe pointer to the flat hash map.
[in]key_val_typethe complete key and value type to be inserted.
Returns
an entry. If Occupied, the entry contains a reference to the key value user type in the table and may be unwrapped. If Vacant the entry contains a reference to the newly inserted entry in the table. If more space is needed but allocation fails or has been forbidden, an insert error is set.
Warning
because this function returns a reference to a user type in the table any subsequent insertions or deletions invalidate this reference.

◆ ccc_fhm_unwrap()

void * ccc_fhm_unwrap ( ccc_fhmap_entry const *  e)

Unwraps the provided entry to obtain a view into the table element.

Parameters
[in]ethe entry from a query to the table via function or macro.
Returns
an view into the table entry if one is present, or NULL.

◆ ccc_fhm_validate()

ccc_tribool ccc_fhm_validate ( ccc_flat_hash_map const *  h)

Validation of invariants for the hash table.

Parameters
[in]hthe table to validate.
Returns
true if all invariants hold, false if corruption occurs. Error if h is NULL.