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!
```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") };
}
See releases for binary downloads.
Usage: citrus <file> [<compiler args…>]
sh
citrus program.c -I./include
The typical workflow is:
C is very weird from Rust perspective. The generated code will be very un-Rust-like. Please don't judge Rust by this :)
size_t
for all lenghts and array indexing (or ssize_t
/ptrdiff_t
if you need negative values). Rust is super picky about this.const
: variables, function arguments, globals. In Rust things are immutable by default.
i
reused throughout a function, (re)define it for each loop.#define
to enum
or const
global variables.inline
functions. For conversion you may even want to undefine assert
, MAX
, and offsetof
.func_int
, func_float
), keep just one version with a unique typedef for the type. You'll be able to replace the typedefed name with a generic parameter later.Replace int
and long
with types of specific size, such as <stdint.h>
's int32_t
, int64_t
or size_t
.
bool
from <stdbool.h>
signed char
only when you really mean to use it for negative numbers.unsigned char
, short
, float
, double
and long long
can be left as-is.In function arguments use arr[]
for arrays, and *ptr
for exactly one element.
arr[i]
yes, ptr+i
no).f(size_t length, arr[static length])
(yes, it's a valid C syntax).__attribute__((nonnull))
to functions that should not be called with NULL
arguments.static
. Syntax of extern C functions will become noisy.for
loops to be in format for(size_t i = start; i < end; i++)
.
while
instead (but avoid do..while
).switch
.
case 1: case 2: … break;
) are OK.var++
in expressions. Use ++var
or put it on its own line. Rust only allows var += 1;
goto
and its labels.Vec
. Rust's fixed-size arrays are PITA.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
export LIBCLANGSTATICPATH=…/clang/build/lib/
export LLVMCONFIGPATH=…/llvm/bin/llvm-config
export LIBCLANGINCLUDEPATH=…/clang/include ```
sh
cargo build