Pnut compiles a reasonably large subset of C99 to human-readable POSIX shell scripts. It can be used to generate portable shell scripts without having to write shell.
Its main uses are:
- As a transpiler to write portable shell scripts in C.
- As a way to bootstrap a compiler written in C with an executable version that is still human readable (See reproducible builds).
Main features:
- No new language to learn -- C code in, shell code out.
- The human-readable shell script is easy to read and understand.
- A runtime library including file I/O and dynamic memory allocations.
- A preprocessor (
#include
,#ifdef
,#define MACRO ...
,#define MACRO_F(x) ...
). - Integrates easily with existing shell scripts.
The examples directory contains many examples. We invite you take a look!
Other than being able to compile itself, Pnut can also compile the Ribbit Virtual Machine which can run a R4RS Scheme Read-eval-print loop directly in shell. See repl.sh for the generated shell script.
Pnut can be distributed as the pnut.sh
shell script, or compiled to executable
code using a C compiler.
To install pnut:
> git clone https://github.com/udem-dlteam/pnut.git
> cd pnut
> sudo make install
This installs both pnut.sh
and pnut
in /usr/local/bin
.
Certain compilation options can be used to change the generated shell script:
-DRT_COMPACT
reduces the size of the runtime library at the cost of reduced I/O performance.-DSH_SAVE_VARS_WITH_SET
reduces the overhead of local variables at the cost of readability. This can reduce the execution time of certain programs by more than 50%.-DSH_INCLUDE_C_CODE
includes the original C code in the generated shell script.
They can be set using make install BUILD_OPT="..."
.
The pnut
compiler takes a C file path as input, and outputs to stdout the
POSIX shell code.
Here's an example of how to compile a C file using Pnut:
> pnut.sh examples/fib.c > fib.sh # Compile fib.c to a shell script
> chmod +x fib.sh # Make the shell script executable
> ./fib.sh # Run the shell script
The #include_shell "{file.sh}"
directive can be used to include shell code in
the generated shell script. This makes it possible to call system utilities from
C code, or to use shell scripts generated by Pnut as a library. See
select-file.c and posix-utils.sh for
how to use this feature.
Because Pnut generates purely POSIX shell code, the generated shell scripts
can be run on any POSIX compliant shell. However, certain shells are faster than
others. For faster scripts, we recommend the use of ksh
, dash
or bash
.
zsh
is also supported but tends to be slower on large programs.
Because Pnut can be distributed as a human-readable shell script (pnut.sh
), it
can serve as the basis for a reproducible build system. With a POSIX compliant
shell, pnut.sh
is sufficiently powerful to compile itself and, in the future,
to bootstrap TCC. Because TCC can be used to compile
GCC, this will make it possible to bootstrap a fully featured build toolchain
from only human-readable source files and a POSIX shell.
Because pnut.sh
cannot support certain C features used by TCC, Pnut features
a native code backend that supports a larger subset of C. We call this
compiler pnut-exe
, and it can be compiled using pnut.sh
. The work to make
pnut-exe
compatible with TCC is ongoing.
Once pnut-exe
supports a large enough subset of C99 to compile TCC, the
following steps will be taken to bootstrap TCC from pnut.sh
:
- Compile
pnut-exe.c
topnut-exe.sh
usingpnut.sh
.pnut-exe.sh
is a shell script that turns C code into machine code. - Compile
pnut-exe.c
topnut-exe
usingpnut-exe.sh
. This version ofpnut-exe
is an executable and is much faster. - Compile TCC using
pnut-exe
.
Unfortunately, certains C constructs don't map nicely to POSIX shell which means:
- No support for floating point numbers and unsigned integers.
goto
andswitch
fallthrough are not supported.- The address of (
&
) operator on local variables is not supported. - Arrays and structures cannot be stack-allocated or passed by value.
- Function pointers and indirect calls.
- All local variable declarations must be at the beginning of a function.
- Arrays and structures cannot be nested in a structure.
You can find my slides from my presentation at the Software Language Engineering
(SLE24) conference and the paper the presentation is based in the doc/
directory.
Pnut is a research project and contributions are welcome. Please open an issue to report any bugs or to discuss new features.
To make sure your changes are good, a good practice is to attempt the bootstrap
of pnut.c
using pnut.sh
. This can be done using the ./bootstrap-pnut-sh.sh
command. Using ksh
, this should take around 30s.