c-raii
An robust high-level Defer, RAII implementation for C89
, automatic memory safety, smarty!
- Synopsis
- There is 1 way to create an smart memory pointer.
- The following malloc/calloc wrapper functions are used to get an raw memory pointer.
- Thereafter, an smart pointer can be use with these raii* functions
- Using
thread local storage
for an default smart pointer, the following functions always available. - Fully automatic memory safety, using
guard/unguarded/guarded
macro.
- Installation
- Contributing
- License
This library has been decoupled from c-coroutine to be independently developed.
In the effort to find uniform naming of terms, various other packages was discovered Except, and exceptions-and-raii-in-c. Choose to settle in on A defer mechanism for C, an upcoming C standard compiler feature. It’s exactly this library’s working design and concepts addressed in c-coroutine.
This library uses an custom version of rpmalloc for malloc/heap allocation, not C11 compiler thread local
dependant, nor C++ focus, removed. The customization is merged with required cthread for C11 thread like emulation.
- The behavior here is as in other languages Go’s defer, Zig’s defer, Swift’s C defer, even Rust has multi defer crates there are other borrow checker issues - A defer discussion.
As a side benefit, just including a single
#include "raii.h"
will make your Linux only application Windows compatible, seework-steal.c
in examples folder, it’s from Complementary Concurrency Programs for course “Linux Kernel Internals”, 2 minor changes, using macromake_atomic
.
All aspect of this library as been incorporated into an c++11 like threading model as outlined in std::async, any thread launched with signature
thrd_for(thread_func, args_for("six", "text & num & another func", 3, worker_func))
has function executed withinguard
block as further described below, all allocations is automatically cleaned up anddefer
statements run.
The C++ version from https://cplusplus.com/reference/future/future/wait/
RAII C89 | C++11 | ||||||||
---|---|---|---|---|---|---|---|---|---|
```c #include "raii.h" // a non-optimized way of checking for prime numbers: void *is_prime(args_t arg) { int i, x = get_arg(arg).integer; for (i = 2; i < x; ++i) if (x % i == 0) return thrd_value(false); return thrd_value(true); } int main(int argc, char **argv) { int prime = 194232491; // call function asynchronously: future *fut = thrd_for(is_prime, &prime); // a status check, use to guarantee any `thrd_get` call will be ready (and not block) // must be part of some external event loop handling routine if (!thrd_is_done(fut)) printf("checking...\n"); printf("\n194232491 "); if (thrd_get(fut).boolean) // blocks and wait for is_prime to return printf("is prime.\n"); else printf("is not prime.\n"); return 0; } ``` |
```c++
// future::wait
#include
Calling g. Printing in g = 0. Printing in g = 1. Printing in g = 2. Printing in g = 3. Panicking! Defer in g = 3. Defer in g = 2. Defer in g = 1. Defer in g = 0. In defer in f Recovered in f = 4 Returned normally from f.## Synopsis **C++** has a concept called [unique_ptr)](https://en.cppreference.com/w/cpp/memory/unique_ptr) where **"a smart pointer that owns and manages another object through a pointer and disposes of that object when the unique_ptr goes out of scope. "** Here too the same process is in effect through an **new** _typedef_ `unique_t` aka `memory_t` _structure_. ```c /* Calls fn (with args as arguments) in separate thread, returning without waiting for the execution of fn to complete. The value returned by fn can be accessed by calling `thrd_get()`. */ C_API future *thrd_for(thrd_func_t fn, void_t args); /* Returns the value of a `future` ~promise~ thread's shared object, If not ready, this function blocks the calling thread and waits until it is ready. */ C_API values_type thrd_get(future *); /* Check status of `future` object state, if `true` indicates thread execution has ended, any call thereafter to `thrd_get` is guaranteed non-blocking. */ C_API bool thrd_is_done(future *); C_API uintptr_t thrd_self(void); C_API raii_values_t *thrd_value(uintptr_t value); /** * Creates an scoped container for arbitrary arguments passing to an single `args` function. * Use `get_args()` or `args_in()` for retrieval. * * @param scope callers context to bind `allocated` arguments to * @param desc format, similar to `printf()`: * * `i` unsigned integer, * * `d` signed integer, * * `l` signed long, * * `z` size_t - max size, * * `c` character, * * `s` string, * * `a` array, * * `x` function, * * `f` double/float, * * `p` void pointer for any arbitrary object * @param arguments indexed by `desc` format order */ C_API args_t raii_args_for(memory_t *scope, const char *desc, ...); C_API args_t args_for(const char *desc, ...); /** * Returns generic union `values_type` of argument, will auto `release/free` * allocated memory when scoped return/exit. * * Must be called at least once to release `allocated` memory. * * @param params arbitrary arguments * @param item index number */ C_API values_type raii_get_args(memory_t *scope, void_t params, int item); C_API values_type get_args(void *params, int item); C_API values_type get_arg(void_t params); ``` ### There is _1 way_ to create an smart memory pointer ```c /* Creates smart memory pointer, this object binds any additional requests to it's lifetime. for use with `malloc_*` `calloc_*` wrapper functions to request/return raw memory. */ C_API unique_t *unique_init(void); ``` > This system use macros `RAII_MALLOC`, `RAII_FREE`, `RAII_REALLOC`, and `RAII_CALLOC`. > If not **defined**, the default **malloc/calloc/realloc/free** are used when expanded. > The expansion thereto points to custom replacement allocation routines **rpmalloc**. ### The following _malloc/calloc_ wrapper functions are used to get an raw memory pointer ```c /* Request/return raw memory of given `size`, using smart memory pointer's lifetime scope handle. DO NOT `free`, will be `RAII_FREE` when scope smart pointer panics/returns/exits. */ C_API void *malloc_by(memory_t *scope, size_t size); /* Request/return raw memory of given `size`, using smart memory pointer's lifetime scope handle. DO NOT `free`, will be `RAII_FREE` when scope smart pointer panics/returns/exits. */ C_API void *calloc_by(memory_t *scope, int count, size_t size); /* Request/return raw memory of given `size`, using smart memory pointer's lifetime scope handle. DO NOT `free`, will be freed with given `func`, when scope smart pointer panics/returns/exits. */ C_API void *malloc_full(memory_t *scope, size_t size, func_t func); /* Request/return raw memory of given `size`, using smart memory pointer's lifetime scope handle. DO NOT `free`, will be freed with given `func`, when scope smart pointer panics/returns/exits. */ C_API void *calloc_full(memory_t *scope, int count, size_t size, func_t func); ``` Note the above functions will **panic/throw** if request fails, is `NULL`, and begin **unwinding**, executing _deferred_ statements. ### Thereafter, an smart pointer can be use with these _raii_* functions ```c /* Defer execution `LIFO` of given function with argument, to the given `scoped smart pointer` lifetime/destruction. */ C_API size_t raii_deferred(memory_t *, func_t, void *); /* Same as `raii_deferred` but allows recover from an Error condition throw/panic, you must call `raii_is_caught` inside function to mark Error condition handled. */ C_API void raii_recover_by(memory_t *, func_t, void *); /* Compare `err` to scoped error condition, will mark exception handled, if `true`. */ C_API bool raii_is_caught(memory_t *scope, const char *err); /* Get scoped error condition string. */ C_API const char *raii_message_by(memory_t *scope); /* Begin `unwinding`, executing given scope smart pointer `raii_deferred` statements. */ C_API void raii_deferred_free(memory_t *); /* Same as `raii_deferred_free`, but also destroy smart pointer. */ C_API void raii_delete(memory_t *ptr); ``` ### Using `thread local storage` for an default smart pointer, the following functions always available ```c /* Request/return raw memory of given `size`, uses current `thread` smart pointer, DO NOT `free`, will be `RAII_FREE` when `raii_deferred_clean` is called. */ C_API void *malloc_default(size_t size); /* Request/return raw memory of given `size`, uses current `thread` smart pointer, DO NOT `free`, will be `RAII_FREE` when `raii_deferred_clean` is called. */ C_API void *calloc_default(int count, size_t size); /* Defer execution `LIFO` of given function with argument, to current `thread` scope lifetime/destruction. */ C_API size_t raii_defer(func_t, void *); /* Same as `raii_defer` but allows recover from an Error condition throw/panic, you must call `raii_caught` inside function to mark Error condition handled. */ C_API void raii_recover(func_t, void *); /* Compare `err` to current error condition, will mark exception handled, if `true`. */ C_API bool raii_caught(const char *err); /* Get current error condition string. */ C_API const char *raii_message(void); /* Begin `unwinding`, executing current `thread` scope `raii_defer` statements. */ C_API void raii_deferred_clean(void); ``` ### Fully automatic memory safety, using `guard/unguarded/guarded` macro The full potently of **RAII** is encapsulated in the `guard` macro. Using `try/catch/catch_any/catch_if/finally/end_try` exception system macros separately will be unnecessary, however see [examples](https://github.com/zelang-dev/c-raii/tree/main/examples) folder for usage. ```c++ try { throw(division_by_zero); printf("never reached\n"); } catch_if { if (caught(bad_alloc)) printf("catch: exception %s (%s:%d) caught\n", err, err_file, err_line); else if (caught(division_by_zero)) printf("catch: exception %s (%s:%d) caught\n", err, err_file, err_line); ex_backtrace(err_backtrace); } finally { if (err) printf("finally: try failed -> %s (%s:%d)\n", err, err_file, err_line); else printf("finally: try succeeded\n"); } end_try; ``` The Planned C11 implementation details still holds here, but `defer` not confined to `guard` block, actual function call. ```c /* Creates an scoped guard section, it replaces `{`. Usage of: `_defer`, `_malloc`, `_calloc`, `_assign_ptr` macros are only valid between these sections. - Use `_return(x);` macro, or `break;` to exit early. - Use `_assign_ptr(var)` macro, to make assignment of block scoped smart pointer. */ #define guard /* This ends an scoped guard section, it replaces `}`. On exit will begin executing deferred functions, return given `result` when done, use `NONE` for no return. */ #define unguarded(result) /* This ends an scoped guard section, it replaces `}`. On exit will begin executing deferred functions. */ #define guarded /* Returns protected raw memory pointer, DO NOT FREE, will `throw/panic` if memory request fails. */ #define _malloc(size) /* Returns protected raw memory pointer, DO NOT FREE, will `throw/panic` if memory request fails. */ #define _calloc(count, size) /* Defer execution `LIFO` of given function with argument, execution begins when current `guard` scope exits or panic/throw. */ #define _defer(func, ptr) /* Compare `err` to scoped error condition, will mark exception handled, if `true`. */ #define _recover(err) /* Compare `err` to scoped error condition, will mark exception handled, if `true`. DO NOT PUT `err` in quote's like "err". */ #define _is_caught(err) /* Get scoped error condition string. */ #define _get_message() /* Stops the ordinary flow of control and begins panicking, throws an exception of given message. */ #define _panic(err) /* Makes a reference assignment of current scoped smart pointer. */ #define _assign_ptr(scope) /* Exit `guarded` section, begin executing deferred functions, return given `value` when done, use `NONE` for no return. */ #define _return(value) ``` The idea way of using this library, is to make a **new** _field_ for `unique_t` into your current _typedef_ **object**, mainly one held throughout application, and setup your wrapper functions to above **raii_** functions. There are also `2 global callback` functions that need to be setup for complete integration. ```c // Currently an wrapper function that set ctx->data, scoped error, and protection state, working on removing need typedef void (*ex_setup_func)(ex_context_t *ctx, const char *ex, const char *panic); // Your wrapper to raii_deferred_free(ctx->data) typedef void (*ex_unwind_func)(void *); ex_setup_func exception_setup_func; ex_unwind_func exception_unwind_func; ``` ## Installation The build system uses **cmake**, by default produces **static** library stored under `built`, and the complete `include` folder is needed. **As cmake project dependency** ```c if(UNIX) set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} -g -D USE_DEBUG") set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -O3 -fomit-frame-pointer -Wno-return-type") endif() if(WIN32) set(CMAKE_C_FLAGS_DEBUG "${CMAKE_C_FLAGS_DEBUG} /D USE_DEBUG") add_definitions(-D_CRT_SECURE_NO_DEPRECATE) add_definitions("/wd4244 /wd4267 /wd4033 /wd4715") endif() add_subdirectory(deps/raii) target_link_libraries(project PUBLIC raii) ``` **Linux** ```shell mkdir build cd build cmake .. -DCMAKE_BUILD_TYPE=Debug/Release -D BUILD_TESTS=OFF -D BUILD_EXAMPLES=OFF # use to not build tests and examples cmake --build . ``` **Windows** ```shell mkdir build cd build cmake .. -D BUILD_TESTS=OFF -D BUILD_EXAMPLES=OFF # use to not build tests and examples cmake --build . --config Debug/Release ``` ## Contributing Contributions are encouraged and welcome; I am always happy to get feedback or pull requests on Github :) Create [Github Issues](https://github.com/zelang-dev/c-raii/issues) for bugs and new features and comment on the ones you are interested in. ## License The MIT License (MIT). Please see [License File](LICENSE.md) for more information. |