rules_jni is a collection of Bazel rules for applications and libraries that mix Java/JVM and C/C++ code via the Java Native Interface (JNI) or the Java Incovation API.
Currently, the rules cover the following use cases for mixed Java and native application or libraries that are currently not covered by the native Bazel rules:
- building a native library for multiple platforms
- bundling a native library in a deploy JAR and loading the correct version at runtime
- access to the OS-specific JNI headers, even when cross-compiling or during multi-platform builds
- using the Java Invocation API to create or attach to a JVM, both with a Bazel-provided JDK and without Bazel
Add the following snippet to your WORKSPACE
:
load("@bazel_tools//tools/build_defs/repo:http.bzl", "http_archive")
http_archive(
name = "fmeum_rules_jni",
sha256 = "9a387a066f683a8aac4d165917dc7fe15ec2a20931894a97e153a9caab6123ca",
strip_prefix = "rules_jni-0.4.0",
url = "https://github.com/fmeum/rules_jni/archive/refs/tags/v0.4.0.tar.gz",
)
load("@fmeum_rules_jni//jni:repositories.bzl", "rules_jni_dependencies")
rules_jni_dependencies()
If you are using Bazel 5 with bzlmod, add the following to your
MODULE.bazel
:
bazel_dep(name = "rules_jni", version = "0.4.0")
# Alternatively, to keep using the repository as @fmeum_rules_jni, use:
bazel_dep(name = "rules_jni", version = "0.4.0", repo_name = "fmeum_rules_jni")
See the documentation for targets, rules and workspace macros provided by rules_jni.
See tests/native_loader
for an example of how to use rules_jni to create, package and use a
native library from Java. An example of a C++ application that starts a JVM and loads and executes Java code from its
Bazel runfiles using the
Java Invocation API can be found in
tests/libjvm_stub
.
rules_jni heavily uses platforms and thus requires at least
Bazel 4.0.0. For advanced use cases such as multi-platform native libraries,
enabling --incompatible_enable_cc_toolchain_resolution
is required.
rules_jni supports the generation of combined Java and C/C++ coverage reports for projects using {cc,java}_jni_library
and @fmeum_rules_jni//jni:libjvm
. This feature currently has the following limitations:
- Only LLVM-based coverage toolchains with
llvm-profdata
are supported. - When using the Java Invocation API to start a JVM from native code,
@fmeum_rules_jni//jni:libjvm
has to be used rather than@fmeum_rules_jni//jni:libjvm_lite
andrules_jni_init
has to be called. - All jars on the classpath of a JVM started with
JNI_CreateJavaVM
have to be deploy jars.
There are also the following known issues with Bazel Java coverage to keep in mind:
java_test
does not collect coverage forcc_binary
targets it executes at runtime (bazelbuild/bazel#15098)- Java coverage is not collected correctly with JDK 16+ (bazelbuild/bazel#15081)
- Coverage is not collected for native code that transitively depends on a
java_jni_library
target (bazelbuild/bazel#15118)
To enable this feature, add the following lines to your project's .bazelrc
:
# Always required.
coverage --combined_report=lcov
coverage --experimental_use_llvm_covmap
coverage --experimental_generate_llvm_lcov
# These flags ensure that the auto-configured C++ toolchain shipped with Bazel
# uses clang and the LLVM coverage tools. They may not be needed or have to be
# adapted if using a custom toolchain.
coverage --repo_env=CC=clang
coverage --repo_env=BAZEL_USE_LLVM_NATIVE_COVERAGE=1
coverage --repo_env=GCOV=llvm-profdata
Then, collect coverage and use genhtml
to generate an HTML report:
bazel coverage //...
genhtml bazel-out/_coverage/_coverage_report.dat
- Jazzer: A coverage-guided, in-process fuzzer for the JVM.