Citrus: Convert C to Rust

This is a tool that helps convert C programs to Rust programs. It transforms C syntax to Rust syntax, but ignores C semantics.

The generated programs will not run, and may even not compile. However, the tool produces readable source code that can be manually refactored into a Rust program.

Note: This is a very early version. There are still obvious bugs and key language features missing. Please contribute bug reports with test cases and, if you can, fixes!

Example

```c void gz_compress(FILE *in, gzFile out) { char buf[BUFLEN]; int len; int err;

for (;;) {
    len = fread(buf, 1, sizeof(buf), in);
    if (ferror(in)) {
        perror("fread");
        exit(1);
    }
    if (len == 0) break;
    if (gzwrite(out, buf, (unsigned)len) != len) error(gzerror(out, &err));
}
fclose(in);
if (gzclose(out) != Z_OK) error("failed gzclose");

} ```

rust unsafe extern "C" fn gz_compress(mut in: *mut FILE, mut out: gzFile) { let mut buf: [i8; 16384]; let mut len; let mut err; loop { len = fread(buf, 1, std::mem::size_of_val(&buf), in); if ferror(in) { perror("fread"); exit(1); } if len == 0 { break } if gzwrite(out, buf, len as c_uint) != len { error(gzerror(out, &err)) }; } fclose(in); if gzclose(out) != 0 { error("failed gzclose") }; }

Installation

See releases for binary downloads.

Usage

Usage: citrus <file> [<compiler args…>]

sh citrus program.c -I./include

The typical workflow is:

  1. Clean up the C code — bad C code makes even worse Rust code.
  2. Convert C syntax to Rust syntax (this one is automated by Citrus!).
  3. Keep rewriting C-isms into Rust-isms until it compiles (some easy translations are also done by Citrus).
  4. Refactor into idiomatic Rust code.

C is very weird from Rust perspective. The generated code will be very un-Rust-like. Please don't judge Rust by this :)

Preparing C code for conversion

Cleanup post conversion

Building the Citrus tool from scratch

Because if the C3 dependency it requires exactly LLVM 4.0 and a corresponding static Clang library (libclang.a + headers). You may need to build Clang from source for this (sorry). The stable C API of clang is not sufficient for the task, so Citrus has to use a fragile C++ clang API, which is not guaranteed to be compatible with anything.

Build/install LLVM 4 and Clang. Set variables to LLVM and Clang locations:

```sh

Must have 'libclang.a'

export LIBCLANGSTATICPATH=…/clang/build/lib/

Path straight to the 'llvm-config' executable

export LLVMCONFIGPATH=…/llvm/bin/llvm-config

Should contain 'clang' and 'clang-c' sub-directories

export LIBCLANGINCLUDEPATH=…/clang/include ```

sh cargo build