blob: 13b1342cefc105335d3dac94d5dd136c29d65bd2 [file] [log] [blame]
// Copyright (c) 2023, 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.policies;
import com.android.tools.r8.graph.AppView;
import com.android.tools.r8.graph.DexProgramClass;
import com.android.tools.r8.shaking.AppInfoWithLiveness;
import com.android.tools.r8.utils.TraversalContinuation;
import com.android.tools.r8.verticalclassmerging.InvokeSpecialToDefaultLibraryMethodUseRegistry;
import com.android.tools.r8.verticalclassmerging.VerticalMergeGroup;
public class NoInterfacesWithInvokeSpecialToDefaultMethodIntoClassPolicy
extends VerticalClassMergerPolicy {
private final AppView<AppInfoWithLiveness> appView;
public NoInterfacesWithInvokeSpecialToDefaultMethodIntoClassPolicy(
AppView<AppInfoWithLiveness> appView) {
this.appView = appView;
}
@Override
public boolean canMerge(VerticalMergeGroup group) {
DexProgramClass sourceClass = group.getSource();
DexProgramClass targetClass = group.getTarget();
// If there is an invoke-special to a default interface method and we are not merging into an
// interface, then abort, since invoke-special to a virtual class method requires desugaring.
if (!sourceClass.isInterface() || targetClass.isInterface()) {
return true;
}
TraversalContinuation<?, ?> result =
sourceClass.traverseProgramMethods(
method -> {
boolean foundInvokeSpecialToDefaultLibraryMethod =
method.registerCodeReferencesWithResult(
new InvokeSpecialToDefaultLibraryMethodUseRegistry(appView, method));
return TraversalContinuation.breakIf(foundInvokeSpecialToDefaultLibraryMethod);
});
return result.shouldContinue();
}
@Override
public String getName() {
return "NoInterfacesWithInvokeSpecialToDefaultMethodIntoClassPolicy";
}
}