5 releases

new 0.1.4 Jan 21, 2025
0.1.3 Jan 12, 2025
0.1.2 Dec 25, 2024
0.1.1 Dec 21, 2024
0.1.0 Dec 21, 2024

#37 in Build Utils

Download history 197/week @ 2024-12-16 192/week @ 2024-12-23 1/week @ 2024-12-30 84/week @ 2025-01-06 48/week @ 2025-01-13

406 downloads per month

Apache-2.0

125KB
1.5K SLoC

logo

cbundl

webpack but for C code.

license-badge tests-badge version-badge


A simple tool that makes self-contained abominations of C code called bundles. It takes many .c and .h files and concatenates figures out the dependencies between them. Then, using patented (not yet) dependency finding algorithms, it arranges the source code in the files in the correct order so as to produce a single .c file that contains everything. In other words, given a bunch of header and implementation files, this tool can produce a single .c file that (hopefully) compiles and works the same way as just compiling each translation unit by itself and then linking them together.

Table of Contents

Usage

The tool operates as a simple preprocessor. Like the C preprocessor, the tool has its own directives that instruct it how to bundle code together. Enough words. Here is an example:

Consider the following C code (don't worry about the // cbundl comments for now):


frob.h

#ifndef _FOO_H
#define _FOO_H

struct frobinator {
  int frob_count;
};

void frobinate(struct frobinator* frob);

#endif

// cbundl: impl=frob.c

frob.c

// cbundl: bundle
#include "frob.h"

#include <stdio.h>

static void update_frob_count(int *frob_count) {
  *frob_count += 1;
}

void frobinate(struct frobinator* frob) {
  printf("frobbed!\n");
  update_frob_count(&frob->frob_count);
}

main.c

#include <stdio.h>

// cbundl: bundle
#include "frob.h"

int main() {
  struct frobinator f = {0};

  for (int i = 0; i < 10; i++)
    frobinate(&f);

  printf("enough...\n");
  return 0;
}

To compile and run this project you would have to do something along the lines of:

$ cc main.c frob.c -o frob

Which would get you a binary. Now let's just assume that you are in a fictional universe where for whatever reason your code will be compiled with a fixed command along the lines of:

$ cc main.c -o frob

Well... that won't work because the linker complains that we did not give it any implementation for frobinate(). We cannot change the compilation command. But we can change what code goes in the compiler. If we insert a pre-processing step on our code before it is even sent to the compiler we could theoretically include the implementation of frobinate() directly in main.c. This way we would end up with a self-contained translation unit which the compiler (and linker) will be happy to assemble into an executable. We can use cbundl for exactly this.

$ cbundl main.c -o final.c

The above command will parse main.c and figure out what dependencies it has. In this example, main.c wants stdio.h and frob.h. Notice the comment above the #include "frob.h". Comments that begin with // cbundl: are called "directives" and give special instructions to cbundl. The directive above frob.h tells cbundl that, to build the final bundle, it needs to include frob.h. The directive at the end of frob.h tells cbundl that the implementation for at least one of the symbols declared by frob.h lives in frob.c. This tells cbundl to include frob.c inside the resulting bundle. That's it. That's the entire tool 👏. The file final.c then contains:

// My amazing header text!

/**
 *
 *                        )                (    (
 *                    ( /(    (           )\ ) )\
 *                (   )\())  ))\   (     (()/(((_)
 *                 )\ ((_)\  /((_)  )\ )   ((_))_
 *               ((_)| |(_)(_))(  _(_/(   _| || |
 *              / _| | '_ \| || || ' \))/ _` || |
 *              \__| |_.__/ \_,_||_||_| \__,_||_|
 *
 *                cbundl X.X.X-release (XXXXXXX)
 *             https://github.com/threadexio/cbundl
 *
 *      Generated at: XXX XX XXX XXX XX:XX:XX (UTC+XX:XX)
 *
 *
 * Use a gun. And if that don't work...
 *                                       use more gun.
 *   - Dr. Dell Conagher
 *
 */

/**
 * bundled from "frob.h"
 */

#ifndef _FOO_H
#define _FOO_H

struct frobinator {
  int frob_count;
};

void frobinate(struct frobinator* frob);

#endif

/**
 * bundled from "main.c"
 */

#include <stdio.h>

int main() {
  struct frobinator f = {0};

  for (int i = 0; i < 10; i++) frobinate(&f);

  printf("enough...\n");
  return 0;
}

/**
 * bundled from "frob.c"
 */

#include <stdio.h>

static void update_frob_count(int* frob_count) {
  *frob_count += 1;
}

void frobinate(struct frobinator* frob) {
  printf("frobbed!\n");
  update_frob_count(&frob->frob_count);
}

The compiler is now happy to make us our binary. 😃 Congratulations, you now know everything about this tool. 👏 And because we were good programmer boys, girls and everything in between, cbundl will also pass the resulting bundle code through a code formatter of our choice (clang-format by default) so its nice and pretty.

Command line arguments
Usage: cbundl [OPTIONS] <path>

Arguments:
  <path>
          Path to the entry source file.

Options:
      --no-config
          Don't load any configuration file. (Overrides `--config`)

      --config <path>
          Specify an alternate configuration file.

          [default: .cbundl.toml cbundl.toml]

      --deterministic[=<boolean>]
          Output a deterministic bundle.

          [possible values: yes, no]

  -o, --output <path>
          Specify where to write the resulting bundle.

          [default: -]

      --no-banner[=<boolean>]
          Don't output the banner at the top of the bundle.

          [possible values: yes, no]

      --no-format[=<boolean>]
          Don't pass the resulting bundle through the formatter.

          [possible values: yes, no]

      --formatter <exe>
          Code formatter. Must format the code from stdin and write it to stdout.

          [default: clang-format]

  -h, --help
          Print help (see a summary with '-h')

  -V, --version
          Print version

Directives

Directives are special single-line comments (//, not /* */) that give instructions to cbundl.

The format of directives is as follows:

// cbundl: <body>

The directive means different things depending on what <body> is. At this time, only 2 different directives exist:

  • bundle
  • impl

bundle

Format: // cbundl: bundle

The bundle directive must always appear exactly above a local #include, without any other comments or code in between. It informs cbundl of a dependency relation between the current file and the #included file. An intuitive way to think about it, is that the current file "wants" the #included file. Any #includes annotated with a bundle directive will not appear in the bundle. Additionally, any #includes not annotated with a bundle directive will be left as-is. This allows you to create a kind of semi-bundle where even the final bundle includes local files. I can't imagine where that would be useful, but you can do it.

impl

Format: // cbundl: impl=<path>

The impl directive, also called an implementation directive, informs cbundl that the current file is implemented by the file specified by <path>. This directive can appear any number of times in the file (if the implementation is split across many other files). It can also appear anywhere in the file, but convention is that impl directives appear only at either the start or the end of the file. Just like #include-ing .c files, using an implementation directive that points to a .h file is generally considered bad practice.

Configuration

cbundl can be configured via a configuration file. The configuration file exposes fine-grained settings for cbundl not available through the command line. By default, cbundl looks for configuration files named .cbundl.toml or cbundl.toml (in that order), though a custom configuration file can be specified via --config. Alternatively, --no-config tells cbundl to ignore any configuration files.

[!NOTE] Command line flags always take priority over the configuration file.

Configuration files for cbundl are written in TOML. An example configuration is given in cbundl.toml.

Workflow

Ok that's all cool and all but how do I integrate it into my workflow? I'm glad you asked. Simple, instead of running just:

$ cc ...

You do:

$ cbundl main.c > bundle.c
$ cc bundle.c -o main

Just make the bundle with cbundl and compile the bundle instead of your source files. You can also write a simple Makefile that does this:

build:
  cbundl main.c > bundle.c
  cc bundle.c -o main

Installation

cbundl provides pre-built release binaries in Releases for all 3 major desktop platforms.

[!NOTE] Those binaries are built in Github Actions. However if you still don't trust the binaries, I don't blame you. Proceed to the Building section.

cargo

If you happen to have cargo installed you can simply do:

$ cargo install cbundl

nix

If you happen to have nix with flakes enabled, you can do:

$ nix run 'github:threadexio/cbundl/master'
# or
$ nix run 'github:threadexio/cbundl/vX.X.X'

[!NOTE] The above will run cbundl from the master branch. You should generally use the second form to pin down exactly which version you want.

Directly without even cloning the repository. Isn't Nix great?

If you want to install cbundl permanently, you can add the flake to your system configuration.

manually

cbundl is a standalone binary. This means you can very easily install it only for your own user. The following will download the latest linux binary from Releases into ~/.bin.

$ mkdir -p ~/.bin
$ curl --proto '=https' --tlsv1.2 -sSfL 'https://github.com/threadexio/cbundl/releases/latest/download/cbundl-linux' -o ~/.bin/cbundl
$ chmod +x ~/.bin/cbundl

You can then add ~/.bin to PATH so you can use the tool like any other command.

  • Temporarily
$ export PATH="$HOME/.bin:$PATH"
  • Permanently
$ echo -e '\nexport PATH="$HOME/.bin:$PATH"\n' >> ~/.profile
$ exec bash

If all goes well, you should then be able to do cbundl --version.

You could however not do any of that and simply download it somewhere and use the full path to run it.

Building

Ironic that a C source code pre-processing tool is not written in C, isn't it? Anyway, as cbundl is written in a modern language called "Rust", you don't have to fiddle with finicky Makefiles or an esoteric cmake setup to get the damn thing to build. Simply do:

$ cargo build # for the debug build
# or
$ cargo build --release # for the release build

Then you can run cbundl through cargo with cargo run or by running it directly from target/debug/cbundl or target/release/cbundl, depending on which you built.

License

Dependencies

~8–15MB
~181K SLoC