blob: 4fa1a731eeee90b5059fe44ea199351e9593a8d0 [file] [log] [blame]
// Copyright (c) 2019, the R8 project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.verticalclassmerging;
import com.android.tools.r8.graph.AppView;
import com.android.tools.r8.graph.DexProgramClass;
import com.android.tools.r8.graph.DexType;
import com.android.tools.r8.graph.classmerging.MergedClasses;
import com.android.tools.r8.shaking.AppInfoWithLiveness;
import com.android.tools.r8.utils.collections.BidirectionalManyToOneHashMap;
import com.android.tools.r8.utils.collections.BidirectionalManyToOneMap;
import com.android.tools.r8.utils.collections.BidirectionalManyToOneRepresentativeHashMap;
import com.android.tools.r8.utils.collections.BidirectionalManyToOneRepresentativeMap;
import com.android.tools.r8.utils.collections.EmptyBidirectionalOneToOneMap;
import com.android.tools.r8.utils.collections.MutableBidirectionalManyToOneRepresentativeMap;
import java.util.Collection;
import java.util.Set;
import java.util.function.BiConsumer;
public class VerticallyMergedClasses implements MergedClasses {
private final BidirectionalManyToOneRepresentativeMap<DexType, DexType> mergedClasses;
private final BidirectionalManyToOneMap<DexType, DexType> mergedInterfacesToClasses;
private final BidirectionalManyToOneMap<DexType, DexType> mergedInterfacesToInterfaces;
public VerticallyMergedClasses(
BidirectionalManyToOneRepresentativeMap<DexType, DexType> mergedClasses,
BidirectionalManyToOneMap<DexType, DexType> mergedInterfacesToClasses,
BidirectionalManyToOneMap<DexType, DexType> mergedInterfacesToInterfaces) {
this.mergedClasses = mergedClasses;
this.mergedInterfacesToClasses = mergedInterfacesToClasses;
this.mergedInterfacesToInterfaces = mergedInterfacesToInterfaces;
}
public static Builder builder() {
return new Builder();
}
public static VerticallyMergedClasses empty() {
EmptyBidirectionalOneToOneMap<DexType, DexType> emptyMap =
new EmptyBidirectionalOneToOneMap<>();
return new VerticallyMergedClasses(emptyMap, emptyMap, emptyMap);
}
@Override
public void forEachMergeGroup(BiConsumer<Set<DexType>, DexType> consumer) {
mergedClasses.forEachManyToOneMapping(consumer);
}
public BidirectionalManyToOneRepresentativeMap<DexType, DexType> getBidirectionalMap() {
return mergedClasses;
}
@Override
public DexType getMergeTargetOrDefault(DexType type, DexType defaultValue) {
return mergedClasses.getOrDefault(type, defaultValue);
}
public Set<DexType> getSources() {
return mergedClasses.keySet();
}
public Set<DexType> getTargets() {
return mergedClasses.values();
}
@Override
public Collection<DexType> getSourcesFor(DexType type) {
return mergedClasses.getKeys(type);
}
public DexType getTargetFor(DexType type) {
assert mergedClasses.containsKey(type);
return mergedClasses.get(type);
}
@Override
public boolean isMergeSource(DexType type) {
return hasBeenMergedIntoSubtype(type);
}
public boolean hasBeenMergedIntoSubtype(DexType type) {
return mergedClasses.containsKey(type);
}
public boolean hasInterfaceBeenMergedIntoClass(DexType interfaceType) {
return mergedInterfacesToClasses.containsKey(interfaceType);
}
public boolean hasInterfaceBeenMergedIntoSubtype(DexType type) {
return mergedInterfacesToClasses.containsKey(type)
|| mergedInterfacesToInterfaces.containsKey(type);
}
public boolean isEmpty() {
return mergedClasses.isEmpty();
}
@Override
public boolean isMergeTarget(DexType type) {
return !getSourcesFor(type).isEmpty();
}
@Override
public boolean verifyAllSourcesPruned(AppView<AppInfoWithLiveness> appView) {
for (DexType source : mergedClasses.keySet()) {
assert appView.appInfo().wasPruned(source)
: "Expected vertically merged class `" + source.toSourceString() + "` to be absent";
}
return true;
}
public static class Builder {
private final MutableBidirectionalManyToOneRepresentativeMap<DexType, DexType> mergedClasses =
BidirectionalManyToOneRepresentativeHashMap.newIdentityHashMap();
private final BidirectionalManyToOneHashMap<DexType, DexType> mergedInterfacesToClasses =
BidirectionalManyToOneHashMap.newIdentityHashMap();
private final BidirectionalManyToOneHashMap<DexType, DexType> mergedInterfacesToInterfaces =
BidirectionalManyToOneHashMap.newIdentityHashMap();
void add(DexProgramClass source, DexProgramClass target) {
mergedClasses.put(source.getType(), target.getType());
if (source.isInterface()) {
if (target.isInterface()) {
mergedInterfacesToInterfaces.put(source.getType(), target.getType());
} else {
mergedInterfacesToClasses.put(source.getType(), target.getType());
}
}
}
Set<DexType> getSourcesFor(DexProgramClass target) {
return mergedClasses.getKeys(target.getType());
}
boolean isMergeSource(DexProgramClass clazz) {
return mergedClasses.containsKey(clazz.getType());
}
boolean isMergeTarget(DexProgramClass clazz) {
return mergedClasses.containsValue(clazz.getType());
}
void merge(VerticallyMergedClasses.Builder other) {
mergedClasses.putAll(other.mergedClasses);
mergedInterfacesToClasses.putAll(other.mergedInterfacesToClasses);
mergedInterfacesToInterfaces.putAll(other.mergedInterfacesToInterfaces);
}
VerticallyMergedClasses build() {
return new VerticallyMergedClasses(
mergedClasses, mergedInterfacesToClasses, mergedInterfacesToInterfaces);
}
}
}