cargo-ndk 4.1.2

Makes building Rust libraries for Android simpler
Documentation
# cargo-ndk - Build Rust code for Android

[<img alt="CI" src="https://github.com/bbqsrc/cargo-ndk/actions/workflows/ci.yml/badge.svg">](https://github.com/bbqsrc/cargo-ndk/actions)
<img alt="Minimum supported Rust version: 1.86" src="https://img.shields.io/badge/MSRV-1.86-informational">
[<img alt="Crates.io Version" src="https://img.shields.io/crates/v/cargo-ndk">](https://lib.rs/crates/cargo-ndk)

This cargo extension handles all the environment configuration needed for successfully building libraries or binaries for Android from a Rust codebase, with support for generating the correct `jniLibs` directory structure.

`cargo-ndk` provides three subcommands to cargo:

- `cargo ndk` — a passthrough for `cargo` applying all the relevant environment variables to ensure a successful build against the NDK
- `cargo ndk-test` — run tests via `adb` automagically
- `cargo ndk-env` — generate sh exports, PowerShell env vars or JSON (i.e. for Visual Studio Code) to make `rust-analyzer` happy, or other unspeakable crimes I hope you keep to yourself

## Table of Contents

- [Installing]#installing
- [Examples]#examples
  - [Building a library for 32-bit and 64-bit ARM systems]#building-a-library-for-32-bit-and-64-bit-arm-systems
- [Usage]#usage
  - [Using `cargo run` binaries via adb]#using-cargo-run-binaries-via-adb
  - [Running your tests on an Android device]#running-your-tests-on-an-android-device
  - [Controlling verbosity]#controlling-verbosity
  - [Environment variable configuration]#environment-variable-configuration
  - [Providing environment variables for C dependencies]#providing-environment-variables-for-c-dependencies
  - [`cargo-ndk`-specific environment variables]#cargo-ndk-specific-environment-variables
  - [Printing the environment]#printing-the-environment
- [Troubleshooting]#troubleshooting
- [Supported hosts]#supported-hosts
- [Local development]#local-development
- [License]#license

## Installing

You'll first need to install all the toolchains you intend to use. Simplest way is with the following:

```
rustup target add \
    aarch64-linux-android \
    armv7-linux-androideabi \
    x86_64-linux-android \
    i686-linux-android
```

Modify as necessary for your use case.

### With `cargo binstall`

To eliminate time wasted building `cargo-ndk` in your CI pipelines, you can use [`cargo-binstall`](https://github.com/cargo-bins/cargo-binstall):

```
cargo binstall cargo-ndk
```

### Building from source

```
cargo install cargo-ndk
```

## Examples

### Building a library for 32-bit and 64-bit ARM systems

```
cargo ndk -t armeabi-v7a -t arm64-v8a -o ./jniLibs build --release
```

This specifies the Android targets to be built (ordinary triples are also supported), the output directory to use for placing the `.so` files in the layout
expected by Android, and then the ordinary flags to be passed to `cargo`.

![Example](./example/example.svg)

## Usage

If you have installed the NDK with Android Studio to its default location, `cargo ndk` will automatically detect
the most recent NDK version and use it. This can be overriden by specifying the path to the NDK root directory in
the `ANDROID_NDK_HOME` environment variable.

### Using `cargo run` binaries via adb

If you want `cargo run` to automatically run via `adb` for Android builds, add a `.cargo/config.toml` to your project with the following content:

```toml
[target.aarch64-linux-android]
runner = "cargo ndk-runner"
```

Add for each target that you are using.

### Running your tests on an Android device

```
cargo ndk-test -t armeabi-v7a
```

This uses `cargo ndk-runner` under the hood to push the binaries to a connected device, and running it in the Android shell.

### Controlling verbosity

Add `-v` or `-vv` as you ordinarily would after the cargo command.

### Environment variable configuration

You can configure `cargo-ndk` using environment variables with the `CARGO_NDK_` prefix:

- `CARGO_NDK_TARGET`: Set default target(s) (comma-separated for multiple targets)
- `CARGO_NDK_PLATFORM`: Set default API platform level  
- `CARGO_NDK_OUTPUT_PATH`: Set default output directory

These can be overridden by command-line arguments.

### Providing environment variables for C dependencies

`cargo-ndk` derives which environment variables to read the same way as the `cc` crate.

### `cargo-ndk`-specific environment variables

These environment variables are exported for use in build scripts and other downstream use cases:

- `CARGO_NDK_ANDROID_PLATFORM`: the Android platform API number as an integer (e.g. `21`)
- `CARGO_NDK_ANDROID_TARGET`: the Android name for the build target (e.g. `armeabi-v7a`)
- `CARGO_NDK_OUTPUT_PATH`: the output path as specified with the `-o` flag
- `CARGO_NDK_SYSROOT_PATH`: path to the sysroot inside the Android NDK
- `CARGO_NDK_SYSROOT_TARGET`: the target name for the files inside the sysroot (differs slightly from the standard LLVM triples)
- `CARGO_NDK_SYSROOT_LIBS_PATH`: path to the libraries inside the sysroot with the given sysroot target (e.g. `$CARGO_NDK_SYSROOT_PATH/usr/lib/$CARGO_NDK_SYSROOT_TARGET`)

Environment variables for bindgen are automatically configured and exported as well.

### Printing the environment

Sometimes you just want the environment variables that `cargo-ndk` configures so you can, say, set up rust-analyzer in VS Code or similar.

If you want to source it into your bash environment:

```
source <(cargo ndk-env)
```

PowerShell:

```
cargo ndk-env --powershell | Invoke-Expression
```

Rust Analyzer and anything else with JSON-based environment handling:

For configuring rust-analyzer, add the `--json` flag and paste the blob into the relevant place in the config.

## Troubleshooting

### The build is complaining that some compiler builtins are missing. What do I do?

Add `--link-builtins` to your `cargo ndk build` command and you should be happy.

### I need to link `libc++_shared.so`

Add `--link-cxx-shared` to your `cargo ndk build`.

## Supported hosts

- Linux
- macOS (`x86_64` and `arm64`)
- Windows

You can also build for Termux or similar by providing the environment variable `CARGO_NDK_ON_ANDROID` at build-time. Please note that this configuration is *not supported*.

## Local development

`git clone` and then install the crate with `cargo`:

```bash
cargo install --path .
```

## License

This project is licensed under either of

 * Apache License, Version 2.0 ([LICENSE-APACHE]LICENSE-APACHE or http://www.apache.org/licenses/LICENSE-2.0)
 * MIT license ([LICENSE-MIT]LICENSE-MIT or http://opensource.org/licenses/MIT)

at your option.