blob: d8007fcb0ee05a76896306a26f75166a757510ee [file] [log] [blame]
// Copyright (c) 2017, 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.naming;
import com.android.tools.r8.graph.AppView;
import com.android.tools.r8.graph.CachedHashValueDexItem;
import com.android.tools.r8.graph.DexClass;
import com.android.tools.r8.graph.DexReference;
import com.android.tools.r8.graph.DexString;
import com.android.tools.r8.graph.DexType;
import com.android.tools.r8.naming.NamingState.InternalState;
import com.android.tools.r8.shaking.AppInfoWithLiveness;
import com.android.tools.r8.shaking.ProguardConfiguration;
import com.google.common.collect.BiMap;
import com.google.common.collect.HashBiMap;
import java.util.IdentityHashMap;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Function;
abstract class MemberNameMinifier<MemberType, StateType extends CachedHashValueDexItem> {
protected final AppView<AppInfoWithLiveness> appView;
protected final List<String> dictionary;
protected final Map<MemberType, DexString> renaming = new IdentityHashMap<>();
protected final NamingState<StateType, ?> globalState;
protected final boolean overloadAggressively;
protected final boolean useApplyMapping;
protected final MemberNamingStrategy strategy;
protected final State minifierState = new State();
// The use of a bidirectional map allows us to map a naming state to the type it represents,
// which is useful for debugging.
private final BiMap<DexType, NamingState<StateType, ?>> states = HashBiMap.create();
MemberNameMinifier(AppView<AppInfoWithLiveness> appView, MemberNamingStrategy strategy) {
ProguardConfiguration proguardConfiguration = appView.options().getProguardConfiguration();
this.appView = appView;
this.dictionary = proguardConfiguration.getObfuscationDictionary();
this.overloadAggressively = proguardConfiguration.isOverloadAggressively();
this.globalState =
NamingState.createRoot(appView.dexItemFactory(), dictionary, getKeyTransform(), strategy);
this.useApplyMapping = proguardConfiguration.hasApplyMappingFile();
this.strategy = strategy;
}
abstract Function<StateType, ?> getKeyTransform();
protected NamingState<StateType, ?> computeStateIfAbsent(
DexType type, Function<DexType, NamingState<StateType, ?>> f) {
return states.computeIfAbsent(type, f);
}
protected boolean alwaysReserveMemberNames(DexClass holder) {
return !useApplyMapping && holder.isNotProgramClass();
}
// A class that provides access to the minification state. An instance of this class is passed
// from the method name minifier to the interface method name minifier.
class State {
DexString getRenaming(MemberType key) {
return renaming.get(key);
}
void putRenaming(MemberType key, DexString value) {
renaming.put(key, value);
}
NamingState<StateType, ?> getState(DexType type) {
return states.get(type);
}
DexType getStateKey(NamingState<StateType, ?> state) {
return states.inverse().get(state);
}
boolean isReservedInGlobalState(DexString name, StateType state) {
return globalState.isReserved(name, state);
}
}
interface MemberNamingStrategy {
DexString next(DexReference source, InternalState internalState);
boolean bypassDictionary();
boolean breakOnNotAvailable(DexReference source, DexString name);
Set<DexReference> noObfuscation();
}
}