blob: 1b0bf7d631b8b422d94f27def106876acbe43bb1 [file] [log] [blame]
// Copyright 2016 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.rules.android;
import static com.google.common.collect.ImmutableList.toImmutableList;
import static com.google.devtools.build.lib.actions.ParameterFile.ParameterFileType.UNQUOTED;
import static com.google.devtools.build.lib.packages.Attribute.attr;
import static com.google.devtools.build.lib.packages.BuildType.LABEL;
import static com.google.devtools.build.lib.packages.BuildType.TRISTATE;
import static com.google.devtools.build.lib.packages.SkylarkProviderIdentifier.forKey;
import static com.google.devtools.build.lib.rules.android.AndroidCommon.getAndroidConfig;
import com.google.common.base.Function;
import com.google.common.base.Functions;
import com.google.common.base.Joiner;
import com.google.common.base.Predicate;
import com.google.common.base.Predicates;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.Sets;
import com.google.common.collect.Streams;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.actions.ExecutionRequirements;
import com.google.devtools.build.lib.actions.MutableActionGraph.ActionConflictException;
import com.google.devtools.build.lib.actions.ParamFileInfo;
import com.google.devtools.build.lib.analysis.ConfiguredAspect;
import com.google.devtools.build.lib.analysis.ConfiguredAspectFactory;
import com.google.devtools.build.lib.analysis.ConfiguredTarget;
import com.google.devtools.build.lib.analysis.RuleContext;
import com.google.devtools.build.lib.analysis.TransitiveInfoProvider;
import com.google.devtools.build.lib.analysis.actions.CustomCommandLine;
import com.google.devtools.build.lib.analysis.actions.CustomCommandLine.VectorArg;
import com.google.devtools.build.lib.analysis.actions.SpawnAction;
import com.google.devtools.build.lib.analysis.config.HostTransition;
import com.google.devtools.build.lib.analysis.configuredtargets.RuleConfiguredTarget.Mode;
import com.google.devtools.build.lib.analysis.platform.ToolchainInfo;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.collect.IterablesChain;
import com.google.devtools.build.lib.collect.nestedset.NestedSet;
import com.google.devtools.build.lib.packages.AspectDefinition;
import com.google.devtools.build.lib.packages.AspectParameters;
import com.google.devtools.build.lib.packages.AttributeMap;
import com.google.devtools.build.lib.packages.NativeAspectClass;
import com.google.devtools.build.lib.packages.NonconfigurableAttributeMapper;
import com.google.devtools.build.lib.packages.Rule;
import com.google.devtools.build.lib.packages.TriState;
import com.google.devtools.build.lib.rules.java.JavaCommon;
import com.google.devtools.build.lib.rules.java.JavaCompilationArgsProvider;
import com.google.devtools.build.lib.rules.java.JavaCompilationInfoProvider;
import com.google.devtools.build.lib.rules.java.JavaInfo;
import com.google.devtools.build.lib.rules.java.JavaRuleOutputJarsProvider;
import com.google.devtools.build.lib.rules.java.JavaRuleOutputJarsProvider.OutputJar;
import com.google.devtools.build.lib.rules.java.proto.JavaProtoAspectCommon;
import com.google.devtools.build.lib.rules.java.proto.JavaProtoLibraryAspectProvider;
import com.google.devtools.build.lib.rules.proto.ProtoInfo;
import com.google.devtools.build.lib.rules.proto.ProtoLangToolchainProvider;
import com.google.devtools.build.lib.skyframe.ConfiguredTargetAndData;
import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
/** Aspect to {@link DexArchiveProvider build .dex Archives} from Jars. */
public final class DexArchiveAspect extends NativeAspectClass implements ConfiguredAspectFactory {
public static final String NAME = "DexArchiveAspect";
/**
* Function that returns a {@link Rule}'s {@code incremental_dexing} attribute for use by this
* aspect. Must be provided when attaching this aspect to a target.
*/
@AutoCodec
public static final Function<Rule, AspectParameters> PARAM_EXTRACTOR =
(Rule rule) -> {
AttributeMap attributes = NonconfigurableAttributeMapper.of(rule);
AspectParameters.Builder result = new AspectParameters.Builder();
TriState incrementalAttr = attributes.get("incremental_dexing", TRISTATE);
result.addAttribute("incremental_dexing", incrementalAttr.name());
return result.build();
};
/**
* Function that limits this aspect to Java 8 desugaring (disabling incremental dexing) when
* attaching this aspect to a target. This is intended for implicit attributes like the stub APKs
* for {@code bazel mobile-install}.
*/
@AutoCodec
static final Function<Rule, AspectParameters> ONLY_DESUGAR_JAVA8 =
(Rule rule) ->
new AspectParameters.Builder()
.addAttribute("incremental_dexing", TriState.NO.name())
.build();
/** Aspect-only label for dexbuidler executable, to avoid name clashes with labels on rules. */
private static final String ASPECT_DEXBUILDER_PREREQ = "$dex_archive_dexbuilder";
/** Aspect-only label for desugaring executable, to avoid name clashes with labels on rules. */
private static final String ASPECT_DESUGAR_PREREQ = "$aspect_desugar";
private static final ImmutableList<String> TRANSITIVE_ATTRIBUTES =
ImmutableList.of(
"deps",
"exports",
"runtime_deps",
":android_sdk",
"aidl_lib", // for the aidl runtime in the android_sdk rule
"$toolchain", // this is _toolchain in Skylark rules (b/78647825)
// To get from proto_library through proto_lang_toolchain rule to proto runtime library.
JavaProtoAspectCommon.LITE_PROTO_TOOLCHAIN_ATTR, "runtime");
private static final FlagMatcher DEXOPTS_SUPPORTED_IN_DEXBUILDER =
new FlagMatcher(
ImmutableList.of("--no-locals", "--no-optimize", "--no-warnings", "--positions"));
private final String toolsRepository;
public DexArchiveAspect(String toolsRepository) {
this.toolsRepository = toolsRepository;
}
@Override
public AspectDefinition getDefinition(AspectParameters params) {
AspectDefinition.Builder result =
new AspectDefinition.Builder(this)
.requireSkylarkProviders(forKey(JavaInfo.PROVIDER.getKey()))
// Latch onto Skylark toolchains in case they have a "runtime" (b/78647825)
.requireSkylarkProviders(forKey(ToolchainInfo.PROVIDER.getKey()))
// For android_sdk rules, where we just want to get at aidl runtime deps.
.requireSkylarkProviders(forKey(AndroidSdkProvider.PROVIDER.getKey()))
.requireSkylarkProviders(forKey(ProtoInfo.PROVIDER.getKey()))
.requireProviderSets(
ImmutableList.of(
// For proto_lang_toolchain rules, where we just want to get at their runtime
// deps.
ImmutableSet.<Class<?>>of(ProtoLangToolchainProvider.class)))
// Parse labels since we don't have RuleDefinitionEnvironment.getLabel like in a rule
.add(
attr(ASPECT_DESUGAR_PREREQ, LABEL)
.cfg(HostTransition.INSTANCE)
.exec()
.value(
Label.parseAbsoluteUnchecked(
toolsRepository + "//tools/android:desugar_java8")))
// Access to --android_sdk so we can stub in a bootclasspath for desugaring if missing
.add(
attr(":dex_archive_android_sdk", LABEL)
.allowedRuleClasses("android_sdk", "filegroup")
.value(
AndroidRuleClasses.getAndroidSdkLabel(
Label.parseAbsoluteUnchecked(
toolsRepository + AndroidRuleClasses.DEFAULT_SDK))))
.requiresConfigurationFragments(AndroidConfiguration.class)
.requireAspectsWithProviders(
ImmutableList.of(ImmutableSet.of(forKey(JavaInfo.PROVIDER.getKey()))))
.requireAspectsWithNativeProviders(JavaProtoLibraryAspectProvider.class);
if (TriState.valueOf(params.getOnlyValueOfAttribute("incremental_dexing")) != TriState.NO) {
// Marginally improves "query2" precision for targets that disable incremental dexing
result.add(
attr(ASPECT_DEXBUILDER_PREREQ, LABEL)
.cfg(HostTransition.INSTANCE)
.exec()
.value(Label.parseAbsoluteUnchecked(toolsRepository + "//tools/android:dexbuilder")));
}
for (String attr : TRANSITIVE_ATTRIBUTES) {
result.propagateAlongAttribute(attr);
}
return result.build();
}
@Override
public ConfiguredAspect create(
ConfiguredTargetAndData ctadBase,
RuleContext ruleContext,
AspectParameters params,
String toolsRepository)
throws InterruptedException, ActionConflictException {
ConfiguredAspect.Builder result = new ConfiguredAspect.Builder(this, params, ruleContext);
Function<Artifact, Artifact> desugaredJars =
desugarJarsIfRequested(ctadBase.getConfiguredTarget(), ruleContext, result);
TriState incrementalAttr =
TriState.valueOf(params.getOnlyValueOfAttribute("incremental_dexing"));
if (incrementalAttr == TriState.NO
|| (!getAndroidConfig(ruleContext).useIncrementalDexing()
&& incrementalAttr == TriState.AUTO)) {
// Dex archives will never be used, so don't bother setting them up.
return result.build();
}
if (JavaCommon.isNeverLink(ruleContext)) {
return result.addProvider(DexArchiveProvider.NEVERLINK).build();
}
DexArchiveProvider.Builder dexArchives =
new DexArchiveProvider.Builder()
.addTransitiveProviders(collectPrerequisites(ruleContext, DexArchiveProvider.class));
Iterable<Artifact> runtimeJars =
getProducedRuntimeJars(ctadBase.getConfiguredTarget(), ruleContext);
if (runtimeJars != null) {
boolean basenameClash = checkBasenameClash(runtimeJars);
Set<Set<String>> aspectDexopts = aspectDexopts(ruleContext);
for (Artifact jar : runtimeJars) {
for (Set<String> incrementalDexopts : aspectDexopts) {
// Since we're potentially dexing the same jar multiple times with different flags, we
// need to write unique artifacts for each flag combination. Here, it is convenient to
// distinguish them by putting the flags that were used for creating the artifacts into
// their filenames.
String uniqueFilename =
(basenameClash ? jar.getRootRelativePathString() : jar.getFilename())
+ Joiner.on("").join(incrementalDexopts)
+ ".dex.zip";
Artifact dexArchive =
createDexArchiveAction(
ruleContext,
ASPECT_DEXBUILDER_PREREQ,
desugaredJars.apply(jar),
incrementalDexopts,
AndroidBinary.getDxArtifact(ruleContext, uniqueFilename));
dexArchives.addDexArchive(incrementalDexopts, dexArchive, jar);
}
}
}
return result.addProvider(dexArchives.build()).build();
}
/**
* Runs Jars in {@link JavaInfo#getDirectRuntimeJars()} through desugaring action if flag is set
* and adds the result to {@code result}. Note that this cannot happen in a separate aspect
* because aspects don't see providers added by other aspects executed on the same target.
*/
private Function<Artifact, Artifact> desugarJarsIfRequested(
ConfiguredTarget base, RuleContext ruleContext, ConfiguredAspect.Builder result) {
if (!getAndroidConfig(ruleContext).desugarJava8()) {
return Functions.identity();
}
Map<Artifact, Artifact> newlyDesugared = new HashMap<>();
if (JavaCommon.isNeverLink(ruleContext)) {
result.addProvider(AndroidRuntimeJarProvider.NEVERLINK);
return Functions.forMap(newlyDesugared);
}
AndroidRuntimeJarProvider.Builder desugaredJars =
new AndroidRuntimeJarProvider.Builder()
.addTransitiveProviders(
collectPrerequisites(ruleContext, AndroidRuntimeJarProvider.class));
if (isProtoLibrary(ruleContext)) {
// TODO(b/33557068): Desugar protos if needed instead of assuming they don't need desugaring
result.addProvider(desugaredJars.build());
return Functions.identity();
}
JavaInfo javaInfo = JavaInfo.getJavaInfo(base);
if (javaInfo != null) {
// These are all transitive hjars of dependencies and hjar of the jar itself
NestedSet<Artifact> compileTimeClasspath =
getJavaCompilationArgsProvider(base, ruleContext).getTransitiveCompileTimeJars();
ImmutableSet.Builder<Artifact> jars = ImmutableSet.builder();
jars.addAll(javaInfo.getDirectRuntimeJars());
// If the target is an android_library, it may be a Starlark android_library in which case
// get the R.jar from the AndroidIdeInfoProvider.
if (isAndroidLibrary(ruleContext)) {
Artifact rJar = getAndroidLibraryRJar(base);
if (rJar != null) {
// TODO(b/124540821): Disable R.jar desugaring (with a flag).
jars.add(rJar);
}
}
// For android_* targets we need to honor their bootclasspath (nicer in general to do so)
ImmutableList<Artifact> bootclasspath = getBootclasspath(base, ruleContext);
ImmutableSet<Artifact> jarsToProcess = jars.build();
boolean basenameClash = checkBasenameClash(jarsToProcess);
for (Artifact jar : jarsToProcess) {
Artifact desugared =
createDesugarAction(
ruleContext, basenameClash, jar, bootclasspath, compileTimeClasspath);
newlyDesugared.put(jar, desugared);
desugaredJars.addDesugaredJar(jar, desugared);
}
}
result.addProvider(desugaredJars.build());
return Functions.forMap(newlyDesugared);
}
private static Iterable<Artifact> getProducedRuntimeJars(
ConfiguredTarget base, RuleContext ruleContext) {
if (isProtoLibrary(ruleContext)) {
if (!ruleContext.getPrerequisites("srcs", Mode.TARGET).isEmpty()) {
JavaRuleOutputJarsProvider outputJarsProvider =
base.getProvider(JavaRuleOutputJarsProvider.class);
if (outputJarsProvider != null) {
return outputJarsProvider
.getOutputJars()
.stream()
.map(OutputJar::getClassJar)
.collect(toImmutableList());
} else {
JavaInfo javaInfo = JavaInfo.getJavaInfo(base);
if (javaInfo != null) {
return javaInfo.getDirectRuntimeJars();
}
}
}
} else {
ImmutableSet.Builder<Artifact> jars = ImmutableSet.builder();
JavaInfo javaInfo = JavaInfo.getJavaInfo(base);
if (javaInfo != null) {
jars.addAll(javaInfo.getDirectRuntimeJars());
}
// The Starlark android_library does not put the R.jar file in the direct runtime deps of the
// JavaInfo, it can only be retrieved from the AndroidIdeInfoProvider.
if (isAndroidLibrary(ruleContext)) {
Artifact rJar = getAndroidLibraryRJar(base);
if (rJar != null) {
jars.add(rJar);
}
}
return jars.build();
}
return null;
}
private static JavaCompilationArgsProvider getJavaCompilationArgsProvider(
ConfiguredTarget base, RuleContext ruleContext) {
JavaCompilationArgsProvider provider =
JavaInfo.getProvider(JavaCompilationArgsProvider.class, base);
if (provider != null) {
return provider;
}
return isProtoLibrary(ruleContext) ? base.getProvider(JavaCompilationArgsProvider.class) : null;
}
private static boolean isAndroidLibrary(RuleContext ruleContext) {
return "android_library".equals(ruleContext.getRule().getRuleClass());
}
private static boolean isProtoLibrary(RuleContext ruleContext) {
return "proto_library".equals(ruleContext.getRule().getRuleClass());
}
private static Artifact getAndroidLibraryRJar(ConfiguredTarget base) {
AndroidIdeInfoProvider provider =
(AndroidIdeInfoProvider) base.get(AndroidIdeInfoProvider.PROVIDER.getKey());
if (provider != null && provider.getResourceJar() != null) {
return provider.getResourceJar().getClassJar();
}
return null;
}
private static boolean checkBasenameClash(Iterable<Artifact> artifacts) {
HashSet<String> seen = new HashSet<>();
for (Artifact artifact : artifacts) {
if (!seen.add(artifact.getFilename())) {
return true;
}
}
return false;
}
private static <T extends TransitiveInfoProvider> Iterable<T> collectPrerequisites(
RuleContext ruleContext, Class<T> classType) {
IterablesChain.Builder<T> result = IterablesChain.builder();
for (String attr : TRANSITIVE_ATTRIBUTES) {
if (ruleContext.attributes().getAttributeType(attr) != null) {
result.add(ruleContext.getPrerequisites(attr, Mode.TARGET, classType));
}
}
return result.build();
}
private static ImmutableList<Artifact> getBootclasspath(
ConfiguredTarget base, RuleContext ruleContext) {
JavaCompilationInfoProvider compilationInfo =
JavaInfo.getProvider(JavaCompilationInfoProvider.class, base);
if (compilationInfo == null || compilationInfo.getBootClasspath().isEmpty()) {
return ImmutableList.of(
ruleContext
.getPrerequisite(":dex_archive_android_sdk", Mode.TARGET, AndroidSdkProvider.PROVIDER)
.getAndroidJar());
}
return compilationInfo.getBootClasspath();
}
private Artifact createDesugarAction(
RuleContext ruleContext,
boolean disambiguateBasenames,
Artifact jar,
ImmutableList<Artifact> bootclasspath,
NestedSet<Artifact> compileTimeClasspath) {
return createDesugarAction(
ruleContext,
ASPECT_DESUGAR_PREREQ,
jar,
bootclasspath,
compileTimeClasspath,
AndroidBinary.getDxArtifact(
ruleContext,
(disambiguateBasenames ? jar.getRootRelativePathString() : jar.getFilename())
+ "_desugared.jar"));
}
/**
* Desugars the given Jar using an executable prerequisite {@code "$desugar"}. Rules calling this
* method must declare the appropriate prerequisite, similar to how {@link #getDefinition} does it
* for {@link DexArchiveAspect} under a different name.
*
* <p>It's useful to have this action separately since callers need to look up classpath and
* bootclasspath in a different way than this aspect does it.
*
* @return the artifact given as {@code result}, which can simplify calling code
*/
static Artifact desugar(
RuleContext ruleContext,
Artifact jar,
ImmutableList<Artifact> bootclasspath,
NestedSet<Artifact> classpath,
Artifact result) {
return createDesugarAction(ruleContext, "$desugar", jar, bootclasspath, classpath, result);
}
private static Artifact createDesugarAction(
RuleContext ruleContext,
String desugarPrereqName,
Artifact jar,
ImmutableList<Artifact> bootclasspath,
NestedSet<Artifact> classpath,
Artifact result) {
CustomCommandLine.Builder args =
new CustomCommandLine.Builder()
.addExecPath("--input", jar)
.addExecPath("--output", result)
.addExecPaths(VectorArg.addBefore("--classpath_entry").each(classpath))
.addExecPaths(VectorArg.addBefore("--bootclasspath_entry").each(bootclasspath));
if (getAndroidConfig(ruleContext).checkDesugarDeps()) {
args.add("--emit_dependency_metadata_as_needed");
}
if (getAndroidConfig(ruleContext).desugarJava8Libs()) {
args.add("--desugar_supported_core_libs");
}
ruleContext.registerAction(
new SpawnAction.Builder()
.useDefaultShellEnvironment()
.setExecutable(ruleContext.getExecutablePrerequisite(desugarPrereqName, Mode.HOST))
.addInput(jar)
.addInputs(bootclasspath)
.addTransitiveInputs(classpath)
.addOutput(result)
.setMnemonic("Desugar")
.setProgressMessage("Desugaring %s for Android", jar.prettyPrint())
.addCommandLine(
// Always use params file, so we don't need to compute command line length first
args.build(), ParamFileInfo.builder(UNQUOTED).setUseAlways(true).build())
.build(ruleContext));
return result;
}
/**
* Creates a dexbuilder action with the given input, output, and flags. Flags must have been
* filtered and normalized to a set that the dexbuilder tool can understand.
*
* @return the artifact given as {@code result}, which can simplify calling code
*/
// Package-private method for use in AndroidBinary
static Artifact createDexArchiveAction(
RuleContext ruleContext,
String dexbuilderPrereq,
Artifact jar,
Set<String> incrementalDexopts,
Artifact dexArchive) {
CustomCommandLine args =
new CustomCommandLine.Builder()
.addExecPath("--input_jar", jar)
.addExecPath("--output_zip", dexArchive)
.addAll(ImmutableList.copyOf(incrementalDexopts))
.build();
SpawnAction.Builder dexbuilder =
new SpawnAction.Builder()
.useDefaultShellEnvironment()
.setExecutable(ruleContext.getExecutablePrerequisite(dexbuilderPrereq, Mode.HOST))
// WorkerSpawnStrategy expects the last argument to be @paramfile
.addInput(jar)
.addOutput(dexArchive)
.setMnemonic("DexBuilder")
.setProgressMessage(
"Dexing %s with applicable dexopts %s", jar.prettyPrint(), incrementalDexopts)
// Always use params file for compatibility with WorkerSpawnStrategy
.addCommandLine(args, ParamFileInfo.builder(UNQUOTED).setUseAlways(true).build());
if (getAndroidConfig(ruleContext).useWorkersWithDexbuilder()) {
dexbuilder.setExecutionInfo(ExecutionRequirements.WORKER_MODE_ENABLED);
}
ruleContext.registerAction(dexbuilder.build(ruleContext));
return dexArchive;
}
private static Set<Set<String>> aspectDexopts(RuleContext ruleContext) {
return Sets.powerSet(
normalizeDexopts(getAndroidConfig(ruleContext).getDexoptsSupportedInIncrementalDexing()));
}
/**
* Derives options to use in incremental dexing actions from the given context and dx flags, where
* the latter typically come from a {@code dexopts} attribute on a top-level target. This method
* only works reliably if the given dexopts were tokenized, e.g., using {@link
* RuleContext#getTokenizedStringListAttr}.
*/
static ImmutableSet<String> incrementalDexopts(
RuleContext ruleContext, Iterable<String> tokenizedDexopts) {
return normalizeDexopts(
Iterables.filter(
tokenizedDexopts,
// dexopts have to match exactly since aspect only creates archives for listed ones
Predicates.in(getAndroidConfig(ruleContext).getDexoptsSupportedInIncrementalDexing())));
}
/**
* Returns the subset of the given dexopts that are blacklisted from using incremental dexing by
* default.
*/
static Iterable<String> blacklistedDexopts(RuleContext ruleContext, List<String> dexopts) {
return Iterables.filter(
dexopts,
new FlagMatcher(
getAndroidConfig(ruleContext).getTargetDexoptsThatPreventIncrementalDexing()));
}
/**
* Derives options to use in DexBuilder actions from the given context and dx flags, where the
* latter typically come from a {@code dexopts} attribute on a top-level target. This should be a
* superset of {@link #incrementalDexopts}.
*/
static ImmutableSet<String> topLevelDexbuilderDexopts(Iterable<String> tokenizedDexopts) {
// We don't need an ordered set but might as well.
return normalizeDexopts(Iterables.filter(tokenizedDexopts, DEXOPTS_SUPPORTED_IN_DEXBUILDER));
}
/**
* Derives options to use in DexFileMerger actions from the given context and dx flags, where the
* latter typically come from a {@code dexopts} attribute on a top-level target.
*/
static ImmutableSet<String> mergerDexopts(
RuleContext ruleContext, Iterable<String> tokenizedDexopts) {
// We don't need an ordered set but might as well. Note we don't need to worry about coverage
// builds since the merger doesn't use --no-locals.
return normalizeDexopts(
Iterables.filter(
tokenizedDexopts,
new FlagMatcher(getAndroidConfig(ruleContext).getDexoptsSupportedInDexMerger())));
}
/**
* Derives options to use in DexFileSharder actions from the given context and dx flags, where the
* latter typically come from a {@code dexopts} attribute on a top-level target.
*/
static ImmutableSet<String> sharderDexopts(
RuleContext ruleContext, Iterable<String> tokenizedDexopts) {
// We don't need an ordered set but might as well. Note we don't need to worry about coverage
// builds since the merger doesn't use --no-locals.
return normalizeDexopts(
Iterables.filter(
tokenizedDexopts,
new FlagMatcher(getAndroidConfig(ruleContext).getDexoptsSupportedInDexSharder())));
}
private static ImmutableSet<String> normalizeDexopts(Iterable<String> tokenizedDexopts) {
// Sort and use ImmutableSet to drop duplicates and get fixed (sorted) order. Fixed order is
// important so we generate one dex archive per set of flag in create() method, regardless of
// how those flags are listed in all the top-level targets being built.
return Streams.stream(tokenizedDexopts)
.map(FlagConverter.DX_TO_DEXBUILDER)
.sorted()
.collect(ImmutableSet.toImmutableSet()); // collector with dedupe
}
private static class FlagMatcher implements Predicate<String> {
private final ImmutableList<String> matching;
FlagMatcher(ImmutableList<String> matching) {
this.matching = matching;
}
@Override
public boolean apply(String input) {
for (String match : matching) {
if (input.contains(match)) {
return true;
}
}
return false;
}
}
private enum FlagConverter implements Function<String, String> {
DX_TO_DEXBUILDER;
@Override
public String apply(String input) {
return input.replace("--no-", "--no");
}
}
}