MOCCA - Make Old C Cexy Again!
CEX is self-contained C language extension, the only dependency is one of gcc/clang compilers. cex.h contains build system, unit test runner, small standard lib and help system.
1. > cd project_dir
2. > gcc/clang ./cex.c -o ./cex (need only once, then cex will rebuild itself)
3. > ./cex --help get info about available commands
- download cex.h
- Make a project directory
mkdir project_dir
cd project_dir
- Make a seed program (NOTE: passing header file is ok)
gcc -D CEX_NEW -x c ./cex.h
clang -D CEX_NEW -x c ./cex.h
- Run cex program for project initilization
./cex
- Now your project is ready to go
./cex test run all
./cex app run myapp
> ./cex --help
Usage:
cex [-D] [-D<ARG1>] [-D<ARG2>] command [options] [args]
CEX language (cexy$) build and project management system
help Search cex.h and project symbols and extract help
process Create CEX namespaces from project source code
new Create new CEX project
config Check project and system environment and config
libfetch Get 3rd party source code via git or install CEX libs
test Test running
app App runner
You may try to get help for commands as well, try `cex process --help`
Use `cex -DFOO -DBAR config` to set project config flags
Use `cex -D config` to reset all project config flags to defaults
CEX is designed as a standalone, single-header programming language with no dependencies other than the GCC/Clang compiler and libc.
Written as a single-header C11 (GNU C) library, CEX is specifically tailored for GCC/Clang compilers. Its mission is to improve C without reinventing a new compiler stack. CEX incorporates modern programming trends while remaining fully compatible with all C tooling.
Though CEX remains unsafe, modern compilers and tools (such as address sanitizers combined with unit tests) help mitigate risks. The language draws inspiration from modern languages, blending their best ideas into a C-compatible form.
The philosophy of CEX revolves around independence and self-containment, minimizing external dependencies. Whenever possible, dependencies are included as source code. CEX embraces the 80/20 principle, prioritizing convenience and developer experience for the majority of use cases (80%). For niche scenarios — whether too large, too small, or requiring extreme performance — a custom solution may be more appropriate.
- Cross-platform, multi-architecture support, big/little endian
- No dependency, single header C programming language less than 20k lines
- Integrated build system - CEX builds itself, no external build system needed!
- New memory management model based on Allocators (temporary memory scopes with auto-free, arenas, etc.)
- New namespacing capabilities for grouping functions / simulating OOP classes
- New error handling model (support of stack traceback on errors, assertions with stack trace (with ASAN))
- Developer experience - unit test runner / code generation / help system included in
cex.h - Code distribution system based on Git and managing dependencies (system libs) with
pkgconf/vcpkg(WIP) - Simple, but powerful standard lib included in
cex.h:- Generic / type-safe dynamic arrays and hashmaps included
- Strings refactored: safe-string functions (copy/formatting), dynamic string buffer (
sbuf), string views/slices (str_s), simple pattern matching engine (wildcard patterns). osnamespace - for running commands, filesystem manipulation, environment variables, path manipulation, platform infoionamespace - cross platform IO support, including helper functions, e.g.io.file.load/save()argparse- convenient argument parsing for CLI tools with built-in commands supportcexy- fancy project management tool and build system.
// CEX has special exception return type that forces the caller to check return type of calling
// function, also it provides support of call stack printing on errors in vanilla C
Exception
cmd_custom_test(u32 argc, char** argv, void* user_ctx)
{
// Let's open temporary memory allocator scope (var name is `_`)
// it will free all allocated memory after any exit from scope (including return or goto)
mem$scope(tmem$, _)
{
e$ret(os.fs.mkpath("tests/build/")); // make directory or return error with traceback
e$assert(os.path.exists("tests/build/")); // evergreen assertion or error with traceback
// auto type variables
var search_pattern = "tests/os_test/*.c";
// Trace with file:<line> + formatting
log$trace("Finding/building simple os apps in %s\n", search_pattern);
// Search all files in the directory by wildcard pattern
// allocate the results (strings) on temp allocator arena `_`
// return dynamic array items type of `char*`
arr$(char*) test_app_src = os.fs.find(search_pattern, false, _);
// for$each works on dynamic, static arrays, and pointer+length
for$each(src, test_app_src)
{
char* tgt_ext = NULL;
char* test_launcher[] = { cexy$debug_cmd }; // CEX macros contain $ in their names
// arr$len() - universal array length getter
// it supports dynamic CEX arrays and static C arrays (i.e. sizeof(arr)/sizeof(arr[0]))
if (arr$len(test_launcher) > 0 && str.eq(test_launcher[0], "wine")) {
// str.fmt() - using allocator to sprintf() format and return new char*
tgt_ext = str.fmt(_, ".%s", "win");
} else {
tgt_ext = str.fmt(_, ".%s", os.platform.to_str(os.platform.current()));
}
// NOTE: cexy is a build system for CEX, it contains utilities for building code
// cexy.target_make() - makes target executable name based on source
char* target = cexy.target_make(src, cexy$build_dir, tgt_ext, _);
// cexy.src_include_changed - parses `src` .c/.h file, finds #include "some.h",
// and checks also if "some.h" is modified
if (!cexy.src_include_changed(target, src, NULL)) {
continue; // target is actual, source is not modified
}
// Launch OS command and get interactive shell
// os.cmd. provides more capabilities for launching subprocesses and grabbing stdout
e$ret(os$cmd(cexy$cc, "-g", "-Wall", "-Wextra", "-o", target, src));
}
}
// CEX provides capabilities for generating namespaces (for user's code too!)
// For example, cexy namespace contains
// cexy.src_changed() - 1st level function
// cexy.app.run() - sub-level function
// cexy.cmd.help() - sub-level function
// cexy.test.create() - sub-level function
return cexy.cmd.simple_test(argc, argv, user_ctx);
}- GCC - 10, 11, 12, 13, 14, 15
- Clang - 13, 14, 15, 16, 17, 18, 19, 20
- MSVC - unsupported, probably never will
- LibC tested - glibc (linux), musl (linux), ucrt/mingw (windows), macos
- Linux - x32 / x64 (glibc, gcc + clang),
- Alpine linux - (libc musl, gcc) on architectures x86_64, x86, aarch64, armhf, armv7, loongarch64, ppc64le, riscv64, and s390x (big-endian)
- Windows (via MSYS2 build) - x64 (mingw64 + clang), libc mscrt/ucrt
- Macos - x64 / arm64 (clang)
CEX is tested on various platforms, compiler versions, sanitizers, and optimization flags, ensuring future compatibility and stability. Sanitizers verify the absence of memory leaks, buffer overflows, and undefined behavior. Additionally, tests with release flags confirm that compiler optimizations do not interfere with the code logic.
| OS / Build type | UBSAN | ASAN | Release -O3 | Release -NDEBUG -O2 |
|---|---|---|---|---|
| Linux Ubuntu 2204 x64 | ✅ | ✅ | ✅ | ✅ |
| Linux Ubuntu 2404 x64 | ✅ | ✅ | ✅ | ✅ |
| Linux Ubuntu 2404 x32 | ✅ | ✅ | ✅ | ✅ |
| Linux Alpine x86_64 | ✅ | ✅ | ✅ | ✅ |
| Linux Alpine x86 | ✅ | ✅ | ||
| Linux Alpine aarch64 | ✅ | ✅ | ||
| Linux Alpine armhf | ✅ | ✅ | ||
| Linux Alpine loongarch64 | ✅ | ✅ | ||
| Linux Alpine ppc64le | ✅ | ✅ | ||
| Linux Alpine riscv64 | ✅ | ✅ | ||
| Linux Alpine s390x | ✅ | ✅ | ||
| Windows 2019 (Win10) x64 | ✅ | ✅ | ✅ | ✅ |
| Windows 2022 (Win10) x64 | ✅ | ✅ | ✅ | ✅ |
| Windows 2025 (Win11) x64 | ✅ | ✅ | ✅ | ✅ |
| MacOS 13 x64 | ✅ | ✅ | ✅ | ✅ |
| MacOS 14 arm64 | ✅ | ✅ | ✅ | ✅ |
| MacOS 15 arm64 | ✅ | ✅ | ✅ | ✅ |
MIT License 2023-2025 (c) Alex Veden