Chromium Docs - Accessing C++ Features In Java (original) (raw)
Checking if a Feature is enabled
In C++, add your base::Feature
to an existing base::android::FeatureMap
in the appropriate layer/component. Then, you can check the enabled state like so:
// FooFeatureMap can check FooFeatures.MY_FEATURE as long as foo_feature_map.cc
// adds kMyFeature
to its base::android::FeatureMap
.
if (FooFeatureMap.getInstance().isEnabled(FooFeatures.MY_FEATURE)) {
// ...
}
If the components or layer does not have a FeatureMap, create a new one:
- In C++, create a new
foo_feature_map.cc
(ex. content_feature_map) with:kFeaturesExposedToJava
array with a pointer to yourbase::Feature
.GetFeatureMap
with a staticbase::android::FeatureMap
initialized withkFeaturesExposedToJava
.JNI_FooFeatureList_GetNativeMap
simply callingGetFeatureMap
.
- In Java, create a
FooFeatureMap.java
class extendingFeatureMap.java
(ex. ContentFeatureMap) with:- A
getInstance()
that returns the singleton instance. - A single
long getNativeMap()
as @NativeMethods. - An
@Override
for the abstractgetNativeMap()
simply callingFooFeatureMapJni.get().getNativeMap()
.
- A
- Still in Java,
FooFeatures.java
with the String constants with the feature names needs to be generated or created.- Auto-generate it by writing a
FooFeatures.java.tmpl
. See instructions below. - If
FooFeatures
cannot be auto-generated, keep the list of String constants with the feature names in aFooFeatures
orFooFeatureList
separate from the pure boilerplateFooFeatureMap
.
- Auto-generate it by writing a
Auto-generating FooFeatureList.java
Accessing C++ base::Features
in Java is implemented via a Python script which analyzes the *_features.cc
file and generates the corresponding Java class, based on a template file. The template file must be specified in the GN target. This outputs Java String constants which represent the name of the base::Feature
.
Usage
- Create a template file (ex.
FooFeatures.java.tmpl
). Change “Copyright 2020” to be whatever the year is at the time of writing (as you would for any other file).
// Copyright 2020 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package org.chromium.foo;
// Be sure to escape any curly braces in your template by doubling as
// follows.
/**
- Contains features that are specific to the foo project.
*/
public final class FooFeatures {{
{NATIVE_FEATURES}
// Prevents instantiation.
private FooFeatures() {{}}
}}
- Add a new build target and add it to the
srcjar_deps
of anandroid_library
target:
if (is_android) {
import("//build/config/android/rules.gni")
}
if (is_android) {
java_cpp_features("java_features_srcjar") {visibility = [ ":*" ] External code should depend on ":foo_java" instead.
sources = [
"//base/android/foo_features.cc",
]
template = "//base/android/java_templates/FooFeatures.java.tmpl"
}
If there's already an android_library target, you can add
java_features_srcjar to that target's srcjar_deps. Otherwise, the best
practice is to create a new android_library just for this target.
android_library("foo_java") {
srcjar_deps = [ ":java_features_srcjar" ]
}
}
3. If you need to expose your flag in WebView, and you created a new android_library
in the previous step, then add a deps
entry to common_java
in //android_webview/BUILD.gn
.
If you don't need to expose a flag in WebView, then skip this and go to the next step.
android_library("common_java") {
...
deps = [
...
"//path/to:foo_java",
...
]
}
4. The generated file out/Default/gen/.../org/chromium/foo/FooFeatures.java
would contain:
// Copyright $YEAR The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package org.chromium.foo;
// Be sure to escape any curly braces in your template by doubling as
// follows.
/**
- Contains features that are specific to the foo project.
*/
public final class FooFeatures {
// This following string constants were inserted by
// java_cpp_features.py
// From
// ../../base/android/foo_features.cc
// Into
// ../../base/android/java_templates/FooFeatures.java.tmpl
// Documentation for the C++ Feature is copied here.
public static final String SOME_FEATURE = "SomeFeature";
// ...snip...
// Prevents instantiation.
private FooFeatures() {}
}
Troubleshooting
The script only supports limited syntaxes for declaring C++ base::Features. You may see an error like the following during compilation:
... org/chromium/foo/FooFeatures.java:41: error: duplicate declaration of field: MY_FEATURE public static final String MY_FEATURE = "MyFeature";
This can happen if you've re-declared a feature for mutually-exclusive build configs (ex. the feature is enabled-by-default for one config, but disabled-by-default for another). Example:
#if defined(...) BASE_FEATURE(kMyFeature, "MyFeature", base::FEATURE_ENABLED_BY_DEFAULT); #else BASE_FEATURE(kMyFeature, "MyFeature", base::FEATURE_DISABLED_BY_DEFAULT); #endif
The java_cpp_features
rule doesn't know how to evaluate C++ preprocessor directives, so it generates two identical Java fields (which is what the compilation error is complaining about). Fortunately, the workaround is fairly simple. Rewrite the definition to only use directives around the enabled state:
BASE_FEATURE(kMyFeature, "MyFeature", #if defined(...) base::FEATURE_ENABLED_BY_DEFAULT #else base::FEATURE_DISABLED_BY_DEFAULT #endif };