JuliaC is a companion to PackageCompiler that streamlines turning Julia code into a native executable, shared library, system image, or intermediate object/bitcode. It provides:
- A CLI app
juliac - A simple library API with explicit "compile → link → bundle" steps
- Optional bundling of
libjulia/stdlibs and artifacts for portable distribution - Optional trimming of IR, metadata, and unreachable code for smaller binaries on Julia 1.12+
Built on top of PackageCompiler.jl.
- Julia 1.12+
- A working C compiler (
clang/gccon macOS/Linux; MSYS2 mingw on Windows)
Install JuliaC as a Julia app:
pkg> app add JuliaCOptional: enable Pkg app shims on your shell PATH so you can run juliac directly:
echo 'export PATH="$HOME/.julia/bin:$PATH"' >> ~/.bashrc # adapt for your shellGiven an app in test/AppProject (see the files in this repository),
compile an executable and produce a self-contained bundle in build/:
juliac \
--output-exe app_test_exe \
--bundle build \
--trim=safe \
--experimental \
test/AppProjectNotes:
--trim[=mode]enables removing unreachable code; on 1.12 prereleases this implies--experimental.- Define
function @main(args::Vector{String})in your package or source file to build an executable. - For simple scripts, a source file may be specified instead of a package directory, but this is not recommended.
julia --project -e "using JuliaC; JuliaC.main(ARGS)" -- \
--output-exe app_test_exe \
--bundle build \
--trim=safe \
--experimental \
test/AppProject--output-exe <name>: Output native executable name (no path). Use--bundleto choose destination directory.--output-lib|--output-sysimage|--output-o|--output-bc <path>: Output path for non-executable artifacts.--project <path>: Project to instantiate/precompile (defaults to active project).--bundle <dir>: Copy required Julia libs/stdlibs and artifacts next to the output; also sets a relative rpath.--trim[=mode]: Enable code trimming (e.g.--trim=safe). Use--trim=noto disable.--compile-ccallable: Exportccallableentrypoints (see C-callable section).--experimental: Forwarded to Julia; required for--trimon some builds.--verbose: Print underlying commands and timing.<file>: The Julia entry file or package to compile (must define@mainfor executables).
using JuliaC
img = ImageRecipe(
output_type = "--output-exe",
file = "test/AppProject",
trim_mode = "safe",
add_ccallables = false,
verbose = true,
)
link = LinkRecipe(
image_recipe = img,
outname = "build/app_test_exe",
rpath = nothing, # set automatically when bundling
)
bun = BundleRecipe(
link_recipe = link,
output_dir = "build", # or `nothing` to skip bundling
)
compile_products(img)
link_products(link)
bundle_products(bun)When --bundle (or BundleRecipe.output_dir) is set, JuliaC:
- Places the executable in
<output_dir>/binand libraries in<output_dir>/liband<output_dir>/lib/julia(Windows: everything under<output_dir>/bin). - Copies required artifacts alongside the bundle.
- Links your output with a relative rpath so the executable finds sibling libs (Unix uses
@loader_path/../libor$ORIGIN/../lib). - On macOS, creates convenience versioned
.dylibsymlinks if missing.
This produces a relocatable directory you can distribute.
On Julia 1.12+, JuliaC can exclude code proven not to be reachable from entry points, reducing binary size:
--trim=safeOn certain builds, --trim requires --experimental (JuliaC will pass it through if needed).
If you pass --compile-ccallable (or set ImageRecipe.add_ccallables = true), JuliaC will export ccallable entrypoints discovered in your code. This is often used when building libraries intended to be called from C or other languages.
- macOS/Linux: requires
clangorgccavailable on PATH - Windows: looks for a MinGW compiler via
LazyArtifactsorJULIA_CC - You can override the compiler with
ENV["JULIA_CC"](e.g.clang/gccpath)
JuliaC builds on PackageCompiler.jl and Julia's own build machinery. Many thanks to their authors and contributors.
Issues and PRs are welcome. Please provide OS, Julia version, and a minimal reproducer when reporting problems.
MIT. See the LICENSE file for details.