blob: 377812746fcbe0d665444b6bbb69a33642e11c06 [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.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkNotNull;
import com.google.common.base.Function;
import com.google.common.collect.HashBasedTable;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.ImmutableTable;
import com.google.common.collect.Table;
import com.google.devtools.build.lib.actions.Artifact;
import com.google.devtools.build.lib.analysis.TransitiveInfoProvider;
import com.google.devtools.build.lib.collect.nestedset.NestedSet;
import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder;
import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.Set;
/**
* Provider of transitively available dex archives corresponding to Jars. A dex archive is a zip of
* {@code .dex} files that each encode exactly one {@code .class} file in an Android-readable form.
* The file names in a dex archive should match the file names in the originating Jar file, except
* with {@code .dex} appended, i.e., {@code <package/for/ClassName[$Inner].class.dex}.
*
* <p>For convenience this class implements a {@link Function} to map from Jars to dex archives if
* available (returns the given Jar otherwise).
*/
@Immutable
public class DexArchiveProvider implements TransitiveInfoProvider {
/**
* Provider that doesn't provide any dex archives, which is what any neverlink target should use.
* It's not strictly necessary to handle neverlink specially, but doing so reduces the amount of
* processing done for targets that won't be used for dexing anyway.
*/
public static final DexArchiveProvider NEVERLINK = new DexArchiveProvider.Builder().build();
/** Builder for {@link DexArchiveProvider}. */
public static class Builder {
private final Table<ImmutableSet<String>, Artifact, Artifact> dexArchives =
HashBasedTable.create();
private final NestedSetBuilder<ImmutableTable<ImmutableSet<String>, Artifact, Artifact>>
transitiveDexArchives = NestedSetBuilder.stableOrder();
public Builder() {}
/**
* Adds all dex archives from the given providers, which is useful to aggregate providers from
* dependencies.
*/
public Builder addTransitiveProviders(Iterable<DexArchiveProvider> providers) {
for (DexArchiveProvider provider : providers) {
transitiveDexArchives.addTransitive(provider.dexArchives);
}
return this;
}
/**
* Adds the given dex archive as a replacement for the given Jar.
*
* @param dexopts
*/
public Builder addDexArchive(Set<String> dexopts, Artifact dexArchive, Artifact dexedJar) {
checkArgument(
dexArchive.getFilename().endsWith(".dex.zip"),
"Doesn't look like a dex archive: %s",
dexArchive);
// Adding this artifact will fail iff dexArchive already appears as the value of another jar.
// It's ok and expected to put the same pair multiple times. Note that ImmutableBiMap fails
// in that situation, which is why we're not using it here.
// It's weird to put a dexedJar that's already in the map with a different value so we fail.
Artifact old =
dexArchives.put(
ImmutableSet.copyOf(dexopts), checkNotNull(dexedJar, "dexedJar"), dexArchive);
checkArgument(
old == null || old.equals(dexArchive),
"We already had mapping %s-%s for dexopts %s, so we don't also need %s",
dexedJar,
old,
dexopts,
dexArchive);
return this;
}
/** Returns the finished {@link DexArchiveProvider}. */
public DexArchiveProvider build() {
return new DexArchiveProvider(
transitiveDexArchives.add(ImmutableTable.copyOf(dexArchives)).build());
}
}
/** Map from Jar artifacts to the corresponding dex archives. */
private final NestedSet<ImmutableTable<ImmutableSet<String>, Artifact, Artifact>> dexArchives;
private DexArchiveProvider(
NestedSet<ImmutableTable<ImmutableSet<String>, Artifact, Artifact>> dexArchives) {
this.dexArchives = dexArchives;
}
/** Returns a flat map from Jars to dex archives transitively produced for the given dexopts. */
public Map<Artifact, Artifact> archivesForDexopts(ImmutableSet<String> dexopts) {
// Can't use ImmutableMap because we can encounter the same key-value pair multiple times.
// Use LinkedHashMap in case someone tries to iterate this map (not the case as of 2/2017).
LinkedHashMap<Artifact, Artifact> result = new LinkedHashMap<>();
for (ImmutableTable<ImmutableSet<String>, Artifact, Artifact> partialMapping :
dexArchives.toList()) {
result.putAll(partialMapping.row(dexopts));
}
return result;
}
@Override
public int hashCode() {
return dexArchives.hashCode();
}
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (obj == null) {
return false;
}
if (getClass() != obj.getClass()) {
return false;
}
DexArchiveProvider other = (DexArchiveProvider) obj;
return dexArchives.equals(other.dexArchives);
}
}