STC is a mature, comprehensive, general purpose container and algorithm library for C99/C11. It has excellent ergonomics and requires virtually no boilerplate code. The library adds many missing features to C, like common data containers, algorithms, and abstractions found in contemporary system languages like Zig, Rust, and C++. Containers and algorithms are templated or generic, which allows for a fully typesafe, compact, and high performance implementation.
Version 6 NEWS
Apologies for the multiple API changes particularly on the coroutine, cspan and cregex modules after V5.1. From upcoming V6.0 release, API is planned to be stable and changes will be made backward compatible.V6.0:
- New powerful V2 coroutines with waitgroups, cancellation and async finalization.
- Fixed #149: improved cregex API and fixed bug on word-boundary match.
- Fixed #143: major container initialization flaw.
- Fixed #146: regression on cstr initialization.
- Fixed #142: regression regarding i_no_clone.
- Fixed #138: general hash function bug.
- Fixed queue/deque shrink_to_fit() and reserve() bugs.
- Fixed i_aux and custom allocations bugs.
- Fixed #136: missing exports in hmap, #137: declare_hash_set()
- Fixed #133: bug in vec/stack _begin() and _rbegin().
- Fixed #129 & #150: Makefile bugs and improvements.
- Fixed #128: bug in cstr_istarts_with().
- Issue #123: Added GNU print format attribute to cstr_from_fmt(), cstr_printf().
- Improved documentation.
V5.1:
- Specifying containers with non-trivial element types can now be done with a single #defineprior to including the container (usingc_keyclass,c_keypro, andc_cmpclassoption traits).
- Users may now define Tas a shorthand fori_type.
- Replaced arc with a new implementation which take up only one pointer. Previous arc is now available as a traits option (c_arc2). The new arc may not be constructed from an object pointer, for that use arc2.
- Updated and fixed bugs in cregex to handle invalid utf8 strings.
- Some breaking changes in cspan API.
- Several other smaller improvements and bug fixes.
V5.0.2:
- Changed c_foreach (...)=>for (c_each(...)), andc_forrange(...)=>for (c_range(...)), etc.
V5.0:
- Added build system/CI with Meson. Makefile provided as well.
- Added support for extending templated containers by #define i_aux <TYPE>.
- Changed ranged for-loop macros to use more natural C-syntax (v5.0.2)
- Added sum type (tagged union), included via algorithm.h
- Added single/multi-dimensional generic span type, with numpy-like slicing.
- Updated coroutines support with structured concurrency and symmetric coroutines.
- Updated coroutines support with proper error handling and error recovery.
- Template parameter Tlets you define container type plusi_keyandi_val(ori_opt) all in one line.
- Template parameters i_keyclassandi_valclassto specify types with_drop()and_clone()functions defined.
- Template parameters i_keyproandi_valproto specifycstr,boxandarctypes (users may also define pro-types).
- hmap now uses Robin Hood hashing (very fast on clang compiler).
- Several new algorithms added, e.g. c_filter(ranges-like),c_shuffle,c_reverse. See also version history for breaking changes in V5.0.
Why use STC?
- A wide set of high performance, generic/templated typesafe container types, including smart pointers and bitsets.
- String type with utf8 support and short string optimization (sso), plus two string-view types.
- Typesafe and ergonomic sum type implementation, aka. tagged union or variant.
- A coroutine implementation with good ergonomics, error handling/recovery and cleanup support.
- Fast, modern regular expressions with full utf8 and a subset of unicode character classes support.
- Ranges algorithms like iota and filter views like take, skip, take-while, skip-while, map.
- Generic algorithms, iterators and loop abstactions. Blazing fast sort, binary search and lower bound.
- Single/multi-dimensional generic span view with arbitrary array dimensions and numpy array-like slicing.
- Abstractions for raw loops, ranged iteration over containers, and generic ranges algorithms. All this reduces the chance of creating bugs, as user code with raw loops and ad-hoc implementation of common algorithms and containers is minimized/eliminated.
- STC is inherently type safe. Essentially, there are no opaque pointers or casting away of type information.
Only where neccesary, generic code will use some macros to do compile-time type-checking before types are casted.
Examples are c_static_assert,c_const_cast,c_safe_castand macros for safe integer type casting.
- Containers and algorithms all use signed integers for indices and sizes, and it encourange to use signed integers for quantities in general (unsigned integers have valid usages as bitsets and in bit operations). This could remove a wide range of bugs related to mixed unsigned-signed calculations and comparisons, which intuitively gives the wrong answer in many cases.
- Tagged unions in C are common, but normally unsafely implemented. Traditionally, it leaves the inactive payload data readily accesible to user code, and there is no general way to ensure that the payload is assigned along with the tag, or that they match. STC sum type is a typesafe version of tagged unions which eliminates all those safety concerns.
- arc - (atomic) reference counted; shared pointer
- box - heap allocated unique pointer`
- cbits - dynamic bitset
- cstr - string type (short string optimized)
- list - forward linked list
- stack - stack type
- vec - vector type
- deque - double-ended queue
- queue - queue type
- pqueue - priority queue
- hashmap - unordered map
- hashset - unordered set
- sortedmap - binary tree map
- sortedset - binary tree set
- cspan - dynamic multi-dimensional (sub)span array view
- csview - string view (non-zero terminated)
- zsview - zero-terminated string view
- Coroutines - ergonomic, portable coroutines
- Regular expressions - Rob Pike's Plan 9 regexp modernized!
- Tagged unions - a.k.a. sum types, variants, discriminating unions
- for-loop abstractions - ranged and on containers
- Misc generic algorithms - incl. fast qsort/binsort/lowerbound
- Random numbers - a very fast PRNG based on SFC64
- Command line argument parser - similar to getopt()
Highlights
- Minimal boilerplate code - Specify only the required template parameters, and leave the rest as defaults.
- Fully type safe - Because of templating, it avoids error-prone casting of container types and elements back and forth from the containers.
- High performance - Unordered maps and sets, queues and deques are significantly faster than the C++ STL containers, the remaining are similar or close to STL in speed (See graph below).
- Fully memory managed - Containers destructs keys/values via default or user supplied drop function. They may be cloned if element types are clonable. Smart pointers (shared and unique) works seamlessly when stored in containers. See arc and box.
- Uniform, easy-to-learn API - For the generic containers and algorithms, simply include the headers. The API and functionality resembles c++ STL or Rust and is fully listed in the docs. Uniform usage across the various containers.
- No signed/unsigned mixing - Unsigned sizes and indices mixed with signed for comparison and calculation is asking for trouble. STC only uses signed numbers in the API for this reason.
- Small footprint - Small source code and generated executables.
- Dual mode compilation - By default it is a header-only library with inline and static methods, but you can easily switch to create a shared library without changing existing source files. Non-generic types, like (utf8) strings are compiled with external linking. one See the installation section.
- No callback functions - All passed template argument functions/macros are directly called from the implementation, no slow callbacks which requires storage.
- Compiles with C++ and C99 - C code can be compiled with C++ (container element types must be POD).
- Pre-declaration - Templated containers may be pre-declared without including the full API/implementation.
- Extendable containers - STC provides a mechanism to wrap containers inside a struct with custom data per instance.
Installation
STC uses meson build system. Make sure to have meson and ninja installed, e.g. as a python pip package from a bash shell:
pip install meson ninja
export LIBRARY_PATH=$LIBRARY_PATH:~/.local/lib
export CPATH=$CPATH:~/.local/include
export CC=gccTo create a build folder and to set the install folder to e.g. ~/.local:
meson setup --buildtype debug build --prefix ~/.local
cd build
ninja
ninja installSTC is mixed "headers-only" / traditional library, i.e the templated container headers (and the sort/lower_bound algorithms) can simply be included - they have no library dependencies. By default, all templated functions are static (many inlined). This is often optimal for both performance and compiled binary size. However, for frequently used container type instances (more than 2-3 TUs), consider creating a separate header file for them, e.g.:
// intvec.h
#ifndef INTVEC_H_
#define INTVEC_H_
#define i_header // header definitions only
#define T intvec, int
#include <stc/vec.h>
#endifSo anyone may use the shared vec-type. Implement the shared functions in one C file (if several containers are shared, you may define STC_IMPLEMENT on top of the file once instead):
// shared.c
#define i_implement // implement the shared intvec.
#include "intvec.h"The non-templated types cstr, csview, cregex, cspan and random, are built as a library (libstc), and is using the meson build system. However, the most common functions in csview and random are inlined. The bitset cbits, the zero-terminated string view zsview and algorthm are all fully inlined and need no linking with the stc-library.
Usage
STC containers have similar functionality to the C++ STL standard containers. All containers except for a few, like cstr and cbits are generic/templated. No type casting is used, so containers are type-safe like templated types in C++. To specify template parameters with STC, you define them as macros prior to including the container, e.g.
#define T Floats, float  // Container type (name, element type)
#include <stc/vec.h>     // "instantiate" the desired container type
#include <stdio.h>
int main(void)
{
    Floats nums = {0};
    Floats_push(&nums, 30.f);
    Floats_push(&nums, 10.f);
    Floats_push(&nums, 20.f);
    for (int i = 0; i < Floats_size(&nums); ++i)
        printf(" %g", nums.data[i]);
    for (c_each(i, Floats, nums))     // Alternative and recommended way to iterate.
        printf(" %g", *i.ref);      // i.ref is a pointer to the current element.
    Floats_drop(&nums); // cleanup memory
}Switching to a different container type, e.g. a sorted set (sset):
[ Run this code ]
#define T Floats, float
#include <stc/sortedset.h> // Use a sorted set instead
#include <stdio.h>
int main(void)
{
    Floats nums = {0};
    Floats_push(&nums, 30.f);
    Floats_push(&nums, 10.f);
    Floats_push(&nums, 20.f);
    // print the numbers (sorted)
    for (c_each(i, Floats, nums))
        printf(" %g", *i.ref);
    Floats_drop(&nums);
}For associative containers and priority queues (hmap, hset, smap, sset, pqueue), comparison/lookup functions
are assumed to be defined. I.e. if they are not specified with template parameters, it assumes default
comparison operators works. To enable search/sort for the remaining containers (stack, vec, queue, deque),
define i_cmp or i_eq and/or i_less for the element type. If the element type is an integral type,
just define i_use_cmp (will use  == and < operators for comparisons).
If an element destructor i_keydrop is defined, i_keyclone function is required.
Alternatively #define i_opt c_no_clone to disable container cloning.
Let's make a vector of vectors, which can be cloned. All of its element vectors will be destroyed when destroying the Vec2D.
[ Run this code ]
#include <stdio.h>
#include <stc/algorithm.h>
#define T Vec, float
#define i_use_cmp        // enable default ==, < and hash operations
#include <stc/vec.h>
#define T Vec2D
#define i_keyclass Vec   // Use i_keyclass when key type has "members" _clone() and _drop().
#define i_use_eq         // vec does not have _cmp(), but it has _eq()
#include <stc/vec.h>
// The above may be written as a one-liners (note the c_-prefix instead of i_):
// #define T Vec, float, (c_use_cmp)
// #include <stc/vec.h>
// #define T Vec2D, Vec, (c_keyclass | c_use_eq)
// #include <stc/vec.h>
int main(void)
{
    Vec* v;
    Vec2D vec_a = {0};                  // All containers in STC can be initialized with {0}.
    v = Vec2D_push(&vec_a, Vec_init()); // push() returns a pointer to the new element in vec.
    Vec_push(v, 10.f);
    Vec_push(v, 20.f);
    v = Vec2D_push(&vec_a, Vec_init());
    Vec_push(v, 30.f);
    Vec_push(v, 40.f);
    Vec2D vec_b = c_make(Vec2D, {
        c_make(Vec, {10.f, 20.f}),
        c_make(Vec, {30.f, 40.f}),
    });
    printf("vec_a == vec_b is %s.\n", Vec2D_eq(&vec_a, &vec_b) ? "true":"false");
    Vec2D clone = Vec2D_clone(vec_a);   // Make a deep-copy of vec
    for (c_each(i, Vec2D, clone))         // Loop through the cloned vector
        for (c_each(j, Vec, *i.ref))
            printf(" %g", *j.ref);
    c_drop(Vec2D, &vec_a, &vec_b, &clone);  // Free all 9 vectors.
}This example uses four different container types:
[ Run this code ]
#include <stdio.h>
#define T hset_int, int
#include <stc/hashset.h>   // unordered/hash set (assume i_key is basic type, uses `==` operator)
struct Point { float x, y; };
// Define cvec_pnt and enable linear search by defining i_eq
#define T vec_pnt, struct Point
#define i_eq(a, b) (a->x == b->x && a->y == b->y)
#include <stc/vec.h>    // vector of struct Point
// enable sort/search. Use native `<` and `==` operators
#define T list_int, int, (c_use_cmp)
#include <stc/list.h>   // singly linked list
#define T smap_int, int, int
#include <stc/sortedmap.h>  // sorted map int => int
int main(void)
{
    // Define four empty containers
    hset_int set = {0};
    vec_pnt vec = {0};
    list_int lst = {0};
    smap_int map = {0};
    c_defer( // Drop the containers at scope exit
        hset_int_drop(&set),
        vec_pnt_drop(&vec),
        list_int_drop(&lst),
        smap_int_drop(&map)
    ){
        enum{N = 5};
        int nums[N] = {10, 20, 30, 40, 50};
        struct Point pts[N] = {{10, 1}, {20, 2}, {30, 3}, {40, 4}, {50, 5}};
        int pairs[N][2] = {{20, 2}, {10, 1}, {30, 3}, {40, 4}, {50, 5}};
        // Add some elements to each container
        for (int i = 0; i < N; ++i) {
            hset_int_insert(&set, nums[i]);
            vec_pnt_push(&vec, pts[i]);
            list_int_push_back(&lst, nums[i]);
            smap_int_insert(&map, pairs[i][0], pairs[i][1]);
        }
        // Find an element in each container
        hset_int_iter i1 = hset_int_find(&set, 20);
        vec_pnt_iter i2 = vec_pnt_find(&vec, (struct Point){20, 2});
        list_int_iter i3 = list_int_find(&lst, 20);
        smap_int_iter i4 = smap_int_find(&map, 20);
        printf("\nFound: %d, (%g, %g), %d, [%d: %d]\n",
                *i1.ref, i2.ref->x, i2.ref->y, *i3.ref,
                i4.ref->first, i4.ref->second);
        // Erase all the elements found
        hset_int_erase_at(&set, i1);
        vec_pnt_erase_at(&vec, i2);
        list_int_erase_at(&lst, i3);
        smap_int_erase_at(&map, i4);
        printf("After erasing the elements found:");
        printf("\n set:");
        for (c_each(i, hset_int, set))
            printf(" %d", *i.ref);
        printf("\n vec:");
        for (c_each(i, vec_pnt, vec))
            printf(" (%g, %g)", i.ref->x, i.ref->y);
        printf("\n lst:");
        for (c_each(i, list_int, lst))
            printf(" %d", *i.ref);
        printf("\n map:");
        for (c_each(i, smap_int, map))
            printf(" [%d: %d]", i.ref->first, i.ref->second);
    }
}Performance
STC is a fast and memory efficient library, and code compiles fast:
Benchmark notes:
- The barchart shows average test times over three compilers: Mingw64 13.1.0, Win-Clang 16.0.5, VC-19-36. CPU: Ryzen 7 5700X.
- Containers uses value types uint64_tand pairs ofuint64_tfor the maps.
- Black bars indicates performance variation between various platforms/compilers.
- Iterations and access are repeated 4 times over n elements.
- access: no entryfor forward_list, deque, and vector because these c++ containers does not have native find().
- deque: insert: n/3 push_front(), n/3 push_back()+pop_front(), n/3 push_back().
- map and unordered map: insert: n/2 random numbers, n/2 sequential numbers. erase: n/2 keys in the map, n/2 random keys.
Some unique features of STC
- Centralized analysis of template parameters. The analyser assigns values to all
non-specified template parameters using meta-programming. You may specify a set of "standard"
template parameters for each container, but as a minimum only one is required: Tori_key(+i_valfor maps). In this case, STC assumes that the elements are of basic types. For non-trivial types, additional template parameters must be given.
- Alternative lookup and insert type. Specify an alternative type to use for
lookup in containers. E.g., containers with STC string elements (cstr) uses const char*as lookup type. Therefore it is not needed to construct (or destroy) acstrin order to lookup a cstr object. Also, one may pass a c-string literal to one of the emplace-functions to implicitly insert a cstr object, i.e.vec_cstr_emplace(&vec, "Hello")as an alternative tovec_cstr_push(&vec, cstr_from("Hello")).
- Standardized container iterators. All containers can be iterated in the same manner, and all use the same element access syntax. The following works for single-element type containers, e.g a linked list:
#define T MyInts, int
#include <stc/list.h>
...
MyInts ints = c_make(MyInts, {3, 5, 9, 7, 2});
for (c_each(it, MyInts, ints)) *it.ref += 42;Naming rules
- 
Naming conventions - Non-templated container names are prefixed by c, e.g.cstr,cbits,cregex.
- Public STC macros and "keywords" are prefixed by c_, e.g.c_each,c_make.
- Template parameter macros are prefixed by i_, e.g.i_key,T.
- All owning containers can be initialized with {0}(alsocstr), i.e. no heap allocation initially.
 
- Non-templated container names are prefixed by 
- 
Common types defined for any container type Cnt: - Cnt
- Cnt_value
- Cnt_raw
- Cnt_iter
 
- 
Functions defined for most container types: - Cnt_init() -> Cnt
- Cnt_with_capacity(isize capacity) -> Cnt
- Cnt_from_n(Cnt_value[], isize n) -> Cnt
- Cnt_reserve(Cnt*, isize capacity)
- Cnt_move(Cnt*) -> Cnt
- Cnt_take(Cnt*, Cnt unowned)
- Cnt_copy(Cnt*, const Cnt* other)
- Cnt_clone(Cnt other) -> Cnt
- Cnt_drop(Cnt*)
- Cnt_value_drop(Cnt_value*)
- Cnt_value_toraw(Cnt_value*) -> Cnt_raw
- Cnt_capacity(Cnt*) -> isize
- Cnt_size(Cnt*) -> isize
- Cnt_is_empty(Cnt*) -> bool
- Cnt_put_n(Cnt*, Cnt_value[], isize n)
- Cnt_push(Cnt*, Cnt_value)
- Cnt_emplace(Cnt*, Cnt_raw)
- Cnt_erase_at(Cnt*, Cnt_iter)
- Cnt_at(Cnt*, isize index OR Cnt_raw) -> Cnt_value*
- Cnt_find(Cnt*, Cnt_raw) -> Cnt_iter
- Cnt_front(Cnt*) -> Cnt_value*
- Cnt_back(Cnt*) -> Cnt_value*
- Cnt_begin(Cnt*) -> Cnt_iter
- Cnt_end(Cnt*) -> Cnt_iter
- Cnt_next(Cnt_iter*)
- Cnt_advance(Cnt_iter, isize n) -> Cnt_iter
 
Defining template parameters
The container template parameters are specified with a #define i_xxxx statement. Each templated
type instantiation requires an #include statement, even if the same container base type was
included earlier. Normally it is sufficient to only define T before including a container:
#define T ContainerType, KeyType[, ValType][, (Options)]Examples of container definitions:
A sortedmap of int => float:
#define T IntfMap, int, float
#include <stc/sortedmap.h>A hashmap of int => string
#define T StrMap, int, cstr, (c_valpro) // cstr is a "pro" type
#include <stc/hashmap.h>A vector of searchable string vectors:
#define T StrVec, cstr, (c_keypro | c_use_eq) // enable vector linear search (find).
#include <stc/vec.h>
#define T StrVecVec, StrVec, (c_keyclass) // container as element has "class" properties
#include <stc/vec.h>The c_keypro and c_keyclass are options, and is specified as the last comma-separated argument
of the T template parameter. They associate the (key) element type name with a set of standard
named "member" functions and assigns them to template parameters. These are then used during the
implementation of the container. NB! Note that the associated/bound "member" functions are only
required to be implemented if the container actually use them. Option flags are boolean properties,
and may be combined with the | operator. Below is a complete list of options that may be
specified for a container:
- c_cmpclass - i_keybinds _cmp(), _eq() and _hash() member names.
- c_keyclass - i_keybinds _clone(), _drop(), _cmp(), _eq(), and _hash() member names.
- c_valclass -  i_valbinds _clone() and _drop() member names for sortedmaps / hashmaps.
- c_keypro - i_keyis a "keyclass" with an associated "cmpclass"-type named KeyType_raw (see below).
- c_valpro - i_valis a "valclass" with an associated "raw"-type named ValType_raw.
- c_use_cmp - enable <comparison on integral types, or the _cmp() member on "pro/class" elements.
- c_use_eq - enable ==on integral types, or the _eq() member on "pro/class" elements.
- c_no_clone - disable clone functionality in container
- c_no_atomic - used with arc type, do simple reference counting instead of atomic.
- c_no_hash - don't enable hash function when "cmpclass" is specified.
- c_declared - container type was predeclared
Bound element "member" functions when c_keyclass / c_valclass / c_cmpclass are specified:
int      KeyType_cmp(const KeyType* x, const KeyType* y);
bool     KeyType_less(const KeyType* x, const KeyType* y);
bool     KeyType_eq(const KeyType* x, const KeyType* y);
size_t   KeyType_hash(const KeyType* kp);
KeyType  KeyType_clone(KeyType k);
void     KeyType_drop(KeyType* kp);
ValType  ValType_clone(ValType v);
void     ValType_drop(ValType* vp);Notes:
- c_use_cmp is only needed for vec, stack, deque, list, as sorting and linear seach are not enabled by default for them. Maps/sets/priority queues enables these by default.
- Comparison uses <and==operators by default whereas when class/pro are specified, it uses the _cmp() member function by default. However, the _cmp() member is also used for equality comparison, so c_use_eq has to be specified in order to use the _eq() member!
- For plain structs (PODs), define i_cmp/i_eq/i_hashmacros when needed, or make it into a "class" by defining required "member" functions, and use the class-options described.
The c_cmpclass's type is equal to the i_key type. However, it is posible to
- i_cmpclassRawType
The assosicated element "member" functions defined from using meta-template parameters may also be
specified/overridden by defining specific template parameters before including the container.
Only i_key is strictly required to be defined for simple non-maps:
- i_keyKeyType - Element type.
- i_keyclassKeyType - Meta template parameter
- i_cmpclassKeyRaw - Meta template parameter (defaults to KeyType)- i_keyfromFunc - Conversion from KeyRaw to KeyType.
- i_keytorawFunc - Conversion from KeyType to KeyRaw.
 
- i_cmpFunc - Three-way comparison of two KeyRaw elements, given as pointers.
- i_lessFunc - Comparison of two KeyRaw elements. Alternative to specifying i_cmp.
- i_eqFunc - Equality comparison of two KeyRaw. Defaults to !i_cmp(x,y).
- i_hashFunc - Hash function taking a KeyRaw pointer. Companion with i_eq. [required] for hmap and hset unless KeyRaw is an integral type (or a struct with no padding space).
- i_keycloneFunc - [required if] i_keydrop is defined (exception for arc, as it shares).
- i_keydropFunc - Destroy key - defaults to empty destructor.
Bound key-element member functions when i_keyclass and/or i_cmpclass parameters are specified:
int      KeyType_cmp(const KeyRaw* rx, const KeyRaw* ry);
bool     KeyType_less(const KeyRaw* rx, const KeyRaw* ry);
bool     KeyType_eq(const KeyRaw* rx, const KeyRaw* ry);
size_t   KeyType_hash(const KeyRaw* rp);
KeyType  KeyType_from(KeyRaw r);
KeyRaw   KeyType_toraw(const KeyType* kp);
KeyType  KeyType_clone(KeyType k);
void     KeyType_drop(KeyType* kp);- i_valValType - [required] for hmap and smap containers.
- i_valrawValRaw - Alternative input type (converted to/from ValType). Defaults to ValType- i_valfromFunc - Conversion from ValRaw to ValType.
- i_valtorawFunc - Conversion from ValType to ValRaw.
 
- i_valcloneFunc - [required if] i_valdrop is defined (exception for arc, as it shares).
- i_valdropFunc - Destroy mapped val - defaults to empty destructor.
Bound mapped-element member functions when i_valraw / i_val is specified:
ValType  ValType_from(ValRaw r);
ValRaw   ValType_toraw(const ValType* vp);
ValType  ValType_clone(ValType v);
void     ValType_drop(ValType* vp);Normally it is simplest to specify the meta-template parameters via the option argument to T,
however, they can be specified as separate template parameters as well. Specifically, i_cmpclass
can be specified as a different type than i_key (c_cmpclass always makes it equal to i_key).
This enables a container to be associated with an additional alternative "raw" input key/val-type,
and one may specify convertion functions between them. Specifically the string, cstr and smart
pointers, box and arc uses this to enhance ergonmics, but every containers may gain efficiency
and usage enhancements from this general built-in mechanism.
- i_cmpclassRawType - Defines i_keyraw and binds i_cmp, i_eq, and i_hash to RawType_cmp(), RawType_eq(), and RawType_hash() comparison functions/macro names. In addition i_keyfrom, i_keytoraw are bound to conversion functions KeyType_from(RawType*) and KeyType_toraw().- If neither i_key nor i_keyclass are defined, i_key will be defined as RawType. In this case, i_keyfrom, i_keytoraw are bound to default pass-through conversion macros.
- Useful alone for containers of views (like csview) - may use c_cmpclass option in that case.
 
- i_keyclassKeyType- Defines i_key and binds i_keyclone, i_keydrop to KeyType_clone() and KeyType_drop() function/macro names.
- Unless i_cmpclassori_keyraware also specified, comparison functions associated with i_cmpclass are also bound.
- Use with container of containers, or in general when the element type has _clone() and _drop() "member" functions.
 
- i_keyproKeyType - Use with "pro"-element types, i.e. library types like cstr, box and arc. It combines the i_keyclass and i_cmpclass properties. Defining i_keypro is equal to defining- i_cmpclass KeyType_raw.
- i_keyclass KeyType
- I.e. i_key,i_keyclone,i_keydrop,i_keyraw,i_keyfrom,i_keytoraw,i_cmp,i_eq,i_hashwill all be textually bound to function names. See the vikings.c example on how to create and instantiate a self-made pro-type.
 
- i_valclassMappedType - Analogous to the i_keyclass, except for comparison and hash funcs.
- i_valproMappedType - Comparison/lookup functions are not relevant for the mapped type, so this defines- i_valraw MappedType_raw (used by emplace and c_make functions only)
- i_valclass MappedType
- I.e. i_val,i_valclone,i_valdrop,i_valraw,i_valfrom,i_valtorawwill all be defined/bound.
 
- i_keyrawRawType - Lookup/emplace-function argument "raw" type. Defaults to i_key.
- i_keyfromFunc(r) - Conversion func from a i_keyraw to return a i_key type.
- i_keytorawFunc(p) - Conversion func from a i_key pointer to a i_keyraw type. [required] if i_keyraw was defined. By default, it returns the dereferenced i_key value.
- i_valrawRawType - Emplace-function argument "raw" type. Defaults to i_val.
- i_valfromFunc(r) - Conversion func from a i_valraw to return a i_val type.
- i_valtorawFunc(p) - Conversion func from a i_val pointer to a i_valraw type.
Specifying comparison parameters
The table below shows the template parameters which must be defined to support element search/lookup and sort for various container type instantiations.
For the containers marked optional, the features are disabled if the template parameter(s) are not defined. Note that the (integral type) columns also applies to "special" key-types, specified with i_keyclass (so not only for true integral types like int or float).
| Container | search (integral type) | sort (integral type) | | | search (struct elem) | sort (struct elem) | optional | 
|---|---|---|---|---|---|---|
| vec, deque, list | i_use_cmp | i_use_cmp | i_eq | i_cmp/i_less | yes | |
| stack | n/a | i_use_cmp | n/a | i_cmp/i_less | yes | |
| box, arc | i_use_cmp | i_use_cmp | i_eq+i_hash | i_cmp/i_less | yes | |
| hmap, hset | n/a | i_eq+i_hash | n/a | no | ||
| smap, sset | i_cmp/i_less | i_cmp/i_less | no | |||
| pqueue | n/a | n/a | i_cmp/i_less | no | ||
| queue | n/a | n/a | n/a | n/a | n/a | 
The emplace methods
STC, like c++ STL, has two sets of methods for adding elements to containers. One set begins with emplace, e.g. vec_X_emplace_back(). This is an ergonimic alternative to vec_X_push_back() when dealing non-trivial container elements, e.g. strings, shared pointers or other elements using dynamic memory or shared resources.
The emplace methods construct / clone the given raw-type element when it is added to the container (specified normally using i_keypro/i_valpro or i_cmpclass or the c_-option variants). In contrast, the non-emplace methods moves the element into the container.
Note: For containers with integral/trivial element types, or when neither i_keyraw/i_valraw is defined,
the emplace functions are not available (or needed), as it can easier lead to mistakes.
| non-emplace: Move | emplace: Embedded copy | Container | 
|---|---|---|
| insert(), push() | emplace() | hmap, smap, hset, sset | 
| insert_or_assign() | emplace_or_assign() | hmap, smap | 
| push() | emplace() | queue, pqueue, stack | 
| push_back(), push() | emplace_back() | deque, list, vec | 
| push_front() | emplace_front() | deque, list | 
Strings are the most commonly used non-trivial data type. STC containers have proper pre-defined definitions for cstr container elements, so they are fail-safe to use both with the emplace and non-emplace methods:
#include <stc/cstr.h>
#define i_keypro cstr  // use i_keypro for "pro" types like cstr, arc, box
#include <stc/vec.h>   // vector of string (cstr)
...
vec_cstr vec = {0};
cstr s = cstr_lit("a string literal");
const char* hello = "Hello";
vec_cstr_push(&vec, cstr_from(hello);    // make a cstr from const char* and move it onto vec
vec_cstr_push(&vec, cstr_clone(s));      // make a cstr clone and move it onto vec
vec_cstr_emplace(&vec, "Yay, literal");  // internally make a cstr from const char*
vec_cstr_emplace(&vec, cstr_clone(s));   // <-- COMPILE ERROR: expects const char*
vec_cstr_emplace(&vec, cstr_str(&s));    // Ok: const char* input type.
cstr_drop(&s)
vec_cstr_drop(&vec);This is made possible because the type configuration may be given an optional conversion/"rawvalue"-type as template parameter, along with a back and forth conversion methods to the container value type.
Rawvalues are primarily beneficial for lookup and map insertions, however the
emplace methods constructs cstr-objects from the rawvalues, but only when required:
hmap_cstr_emplace(&map, "Hello", "world");
// Two cstr-objects were constructed by emplace
hmap_cstr_emplace(&map, "Hello", "again");
// No cstr was constructed because "Hello" was already in the map.
hmap_cstr_emplace_or_assign(&map, "Hello", "there");
// Only cstr_lit("there") constructed. "world" was destructed and replaced.
hmap_cstr_insert(&map, cstr_lit("Hello"), cstr_lit("you"));
// Two cstr's constructed outside call, but both destructed by insert
// because "Hello" existed. No mem-leak but less efficient.
it = hmap_cstr_find(&map, "Hello");
// No cstr constructed for lookup, although keys are cstr-type.Apart from strings, maps and sets are normally used with trivial value types. However, the last example on the hmap page demonstrates how to specify a map with non-trivial keys.
The erase methods
| Name | Description | Container | 
|---|---|---|
| erase() | key based | smap, sset, hmap, hset, cstr | 
| erase_at() | iterator based | smap, sset, hmap, hset, vec, deque, list | 
| erase_range() | iterator based | smap, sset, vec, deque, list | 
| erase_n() | index based | vec, deque, cstr | 
| remove() | remove all matching values | list | 
User-defined container type name
Define T and/or i_key:
// #define T MyVec, int // shorthand
#define T MyVec
#define i_key int
#include <stc/vec.h>
MyVec vec = {0};
MyVec_push(&vec, 42);
...Pre-declarations
Pre-declare templated container in header file. The container can then e.g. be a member of a struct defined in a header file.
- If the container will use an auxiliary member the with i_aux AuxTypeparameter, the declaration must also add it as the last argument:declare_vec_aux(VecType, Element, AuxType).
- Up to, but not including C23, a (c_declared)option must be specified when defining the container. See example below.
// Dataset.h
#ifndef Dataset_H_
#define Dataset_H_
#include <stc/types.h>   // include various container data structure templates
// declare PointVec as a vec. Also struct Point may be incomplete/undeclared.
declare_vec(PointVec, struct Point);
typedef struct Dataset {
    PointVec vertices;
    PointVec colors;
} Dataset;
void Dataset_drop(Dataset* self);
...
#endifDefine and use the "private" container in the c-file:
// Dataset.c
#include "Dataset.h"
#include "Point.h"      // struct Point must be defined here.
#define T PointVec, struct Point, (c_declared) // Was pre-declared.
#include <stc/vec.h>    // Implements PointVec with static linking by default
...Per container-instance customization
Sometimes it is useful to extend a container type to store extra data, e.g. a comparison
or allocator function pointer or a context which the function pointers can use. Most
libraries solve this by adding an opaque pointer (void*) or function pointer(s) into
the data structure for the user to manage. Because most containers are templated,
an auxiliary template parameter, i_aux may be defined to extend the container with
typesafe custom attributes.
The example below shows how to customize containers to work with PostgreSQL memory management.
It adds a MemoryContext to each container by defining the i_aux template parameter.
i_aux may define a struct on the fly, or refer to an already defined type.
Note that pgs_realloc and pgs_free is also passed the
allocated size of the given pointer, unlike standard realloc and free.
self->aux is accessible from the following template parameters / container combinations:
- i_allocator: all containers
- i_eq: all containers
- i_cmp,- i_less: all containers except hmap and hset
- i_hash: hmap and hset
// pgs_alloc.h
#define pgs_malloc(sz) MemoryContextAlloc(self->aux.memctx, sz)
#define pgs_calloc(n, sz) MemoryContextAllocZero(self->aux.memctx, (n)*(sz))
#define pgs_realloc(p, old_sz, sz) (p ? repalloc(p, sz) : pgs_malloc(sz))
#define pgs_free(p, sz) (p ? pfree(p) : (void)0) // pfree/repalloc does not accept NULL.
#define i_aux struct { MemoryContext memctx; }
#define i_allocator pgs
#define i_no_cloneUsage is straight forward:
#define T IMap, int, int
#include "pgs_alloc.h"
#include <stc/sortedmap.h>
void maptest()
{
    IMap map = {.aux={
        AllocSetContextCreate(CurrentMemoryContext, "MapContext", ALLOCSET_DEFAULT_SIZES)
    }};
    for (c_range(i, 1, 16))
        IMap_insert(&map, i*i, i); // uses pgs_realloc()
    for (c_each(i, IMap, map))
        printf("%d:%d ", i.ref->first, i.ref->second);
    IMap_drop(&map); // uses psg_free()
    MemoryContextDelete(map.aux.memctx);
}Another example is to sort struct elements by the active field and reverse flag:
[ Run this code ]
#include <stdio.h>
#include <time.h>
#include <stc/cstr.h>
#include <c11/fmt.h>
typedef struct {
    cstr fileName;
    cstr directory;
    isize size;
    time_t lastWriteTime;
}  FileMetaData;
enum FMDActive {FMD_fileName, FMD_directory, FMD_size, FMD_lastWriteTime, FMD_LAST};
typedef struct { enum FMDActive activeField; bool reverse; } FMDVectorSorting;
int FileMetaData_cmp(const FMDVectorSorting*, const FileMetaData*, const FileMetaData*);
void FileMetaData_drop(FileMetaData*);
#define T FMDVector, FileMetaData, (c_keyclass | c_no_clone)
#define i_aux FMDVectorSorting
#define i_cmp(x, y) FileMetaData_cmp(&self->aux, x, y)
#include <stc/stack.h>
// --------------
int FileMetaData_cmp(const FMDVectorSorting* aux, const FileMetaData* a, const FileMetaData* b) {
    int dir = aux->reverse ? -1 : 1;
    switch (aux->activeField) {
        case FMD_fileName: return dir*cstr_cmp(&a->fileName, &b->fileName);
        case FMD_directory: return dir*cstr_cmp(&a->directory, &b->directory);
        case FMD_size: return dir*c_default_cmp(&a->size, &b->size);
        case FMD_lastWriteTime: return dir*c_default_cmp(&a->lastWriteTime, &b->lastWriteTime);
        default:;
    }
    return 0;
}
void FileMetaData_drop(FileMetaData* fmd) {
    cstr_drop(&fmd->fileName);
    cstr_drop(&fmd->directory);
}
int main(void) {
    FMDVector vec = c_make(FMDVector, {
        {cstr_from("WScript.cpp"), cstr_from("code/unix"), 3624, 123567},
        {cstr_from("CanvasBackground.cpp"), cstr_from("code/unix/canvas"), 38273, 12398},
        {cstr_from("Brush_test.cpp"), cstr_from("code/tests"), 67236, 7823},
    });
    vec.aux.reverse = true;
    for (c_range_t(enum FMDActive, field, FMD_LAST)) {
        vec.aux.activeField = field;
        FMDVector_sort(&vec);
        for (c_each(i, FMDVector, vec)) {
            fmt_println("{:30}{:30}{:10}{:10}",
                        cstr_str(&i.ref->fileName), cstr_str(&i.ref->directory),
                        i.ref->size, i.ref->lastWriteTime);
        }
        puts("");
    }
    FMDVector_drop(&vec);
}Extended templated containers (advanced)
It is possible to extend the functionality of templated container by adding functions to them,
while having all the input and derived/defaulted template parameters available. Below we add
an alias function _len() for _size() in a new file "extvec.h", which can be used like stc/vec.h:
// extvec.h
#define i_extend
#include <stc/vec.h>
STC_INLINE isize _c_MEMB(_len)(Self* self) {
    return _c_MEMB(_size)(self);
}
#include <stc/sys/finalize.h>Memory efficiency
STC is generally very memory efficient. Memory usage for the different containers:
- cstr, vec, stack, pqueue: 1 pointer, 2 isize + memory for elements.
- csview, 1 pointer, 1 isize. Does not own data!
- cspan, 1 pointer and 2 * dimension * int32_t. Does not own data!
- list: Type size: 1 pointer. Each node allocates a struct to store its value and a next pointer.
- deque, queue: Type size: 2 pointers, 2 isize. Otherwise like vec.
- hmap/hset: Type size: 2 pointers, 2 int32_t (default). hmap uses one table of keys+value, and one table of precomputed hash-value/used bucket, which occupies only one byte per bucket. The closed hashing has a default max load factor of 85%, and hash table scales by 1.5x when reaching that.
- smap/sset: Type size: 1 pointer. smap manages its own array of tree-nodes for allocation efficiency. Each node uses two 32-bit ints for child nodes, and one byte for level, but has no parent node.
- arc: Type size: 1 pointer, 1 long for the shared reference counter + memory for the shared element.
- arc2: Type size: 2 pointers, 1 long for the shared reference counter + memory for the shared element.
- box: Type size: 1 pointer + memory for the pointed-to element.
Version history
- This is a major new version, with serveral breaking changes compared to 4.3
- Some API changes in cregex.
- Some API changes in cstrandcsview.
- Renamed czsview type to zsview, some API changes.
- Renamed all member Container_empty() functions to Container_is_empty().
- Changed API in randomnumbers.
- c_init renamed to c_make
- c_forlist renamed to c_foritems
- c_forpair replaced by c_each_kv(changed API).
- Renamed all functions stc_<xxxx>() to c_<xxxx>()in common.h.
- c_SVFMT(sv) renamed tp c_svfmt(sv)
- c_SVARG(sv) renamed tp c_svarg(sv)
- Renamed coroutine cco_yield() to "keyword" cco_yield.
- Swapped 2nd and 3rd argument in c_fortoken()to make it consistent with all otherc_for*(), i.e, input object is third last.
- New header vec.hrenamed from cvec.h
- New header deque.hrenamed from cdeq.h
- New header list.hrenamed from clist.h
- New header stack.hrenamed from cstack.h
- New header queue.hrenamed from cqueue.h
- New header pqueue.hrenamed from cpque.h
- New header hmap.hrenamed from cmap.h
- New header hset.hrenamed from cset.h
- New header smap.hrenamed from csmap.h
- New header sset.hrenamed from csset.h
- New header zsview.hrenamed from czview.h
- New header random.hrenamed from crand.h
- New header types.hrenamed from forward.h
 
- Some API changes in 
- Breaking changes:
- cstr and csview now uses shared linking by default. Implement by either defining i_implementori_staticbefore including.
- Renamed "stc/calgo.h" => "stc/algorithm.h"
- Moved "stc/algo/coroutine.h" => "stc/coroutine.h"- Much improved with some new API and added features.
 
- Removed deprecated "stc/crandom.h". Use "stc/random.h"with the new API.- Reverted names _unif and _norm back to _uniformand_normal.
 
- Reverted names _unif and _norm back to 
- Removed default comparison for list, vec and deque:
- Define i_use_cmpto enable comparison for built-in i_key types (<, ==).
- Use of i_keyclassstill expects comparison functions to be defined.
 
- Define 
- Renamed input enum flags for cregex-functions.
 
- cstr and csview now uses shared linking by default. Implement by either defining 
- cspan: Added column-major order (fortran) multidimensional spans and transposed views (changed representation of strides).
- All new faster and smaller queue and deque implementations, using a circular buffer.
- Renamed i_extern => i_import(i_extern deprecated).- Define i_importbefore#include <stc/cstr.h>will also define full utf8 case conversions.
- Define i_importbefore#include <stc/cregex.h>will also define cstr + utf8 tables.
 
- Define 
- Renamed c_make() => c_make() macro for initializing containers with element lists. c_make deprecated.
- Removed deprecated uppercase flow-control macro names.
- Other smaller additions, bug fixes and improved documentation.
- New home! And online single headers for https://godbolt.org
- Library: https://github.com/stclib/STC
- Headers, e.g. https://raw.githubusercontent.com/stclib/stcsingle/main/stc/vec.h
 
- Much improved documentation
- Added Coroutines + documentation
- Added new random.h API & header. Old crandom.h is deprecated.
- Added c_const_cast()typesafe macro.
- Removed RAII macros usage from examples
- Renamed c_flt_count(i) => c_flt_counter(i)
- Renamed c_flt_last(i) => c_flt_getcount(i)
- Renamed c_ARRAYLEN() => c_countof()
- Removed deprecated c_ARGSV(). Use c_svarg()
- Removed c_PAIR
Major changes:
- A new exciting cspan single/multi-dimensional array view (with numpy-like slicing).
- Signed sizes and indices for all containers. See C++ Core Guidelines by Stroustrup/Sutter: ES.100, ES.102, ES.106, and ES.107.
- Customizable allocator per templated container type.
- Updates on cregex with several new unicode character classes.
- Algorithms:
- crange - similar to boost::irange integer range generator.
- c_forfilter - ranges-like view filtering.
- quicksort - fast quicksort with custom inline comparison.
 
- Renamed c_ARGSV()=>c_svarg(): csview print arg. Notec_sv()is shorthand for csview_from().
- Support for uppercase flow-control macro names in common.h.
- Some API changes in cregex and cstr.
- Create single header container versions with python script.
- Added cregex with documentation - powerful regular expressions.
- Added: c_forfilter: container iteration with "piped" filtering using && operator. 4 built-in filters.
- Added: crange: number generator type, which can be iterated (e.g. with c_forfilter).
- Added back coption - command line argument parsing.
- New + renamed loop iteration/scope macros:
- c_foritems: macro replacing c_forarray and c_apply. Iterate a compound literal list.
 
- Updated cstr, now always takes self as pointer, like all containers except csview.
- Updated vec, deque, changed *_range*function names.