blob: 6057fab5966641aafad794687c2d0e4395ef61ed [file] [log] [blame]
// Copyright (c) 2024, 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.apimodel;
import static com.android.tools.r8.apimodel.ApiModelingTestHelper.verifyThat;
import static com.android.tools.r8.utils.codeinspector.Matchers.isAbsent;
import static org.hamcrest.MatcherAssert.assertThat;
import com.android.tools.r8.CompilationMode;
import com.android.tools.r8.NeverInline;
import com.android.tools.r8.TestBase;
import com.android.tools.r8.TestCompileResult;
import com.android.tools.r8.TestCompilerBuilder;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.TestParametersCollection;
import com.android.tools.r8.apimodel.extension.ExtensionApiLibraryClass;
import com.android.tools.r8.apimodel.extension.ExtensionApiLibraryInterface;
import com.android.tools.r8.apimodel.extension.ExtensionApiLibraryInterfaceImpl;
import com.android.tools.r8.apimodel.extension.ExtensionApiLibraryInterfaceProvider;
import com.android.tools.r8.utils.StringUtils;
import com.android.tools.r8.utils.codeinspector.CodeInspector;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameter;
import org.junit.runners.Parameterized.Parameters;
@RunWith(Parameterized.class)
public class ApiModelOutlineMethodExtensionApiTest extends TestBase {
private static final String EXPECTED_OUTPUT =
StringUtils.lines(
"ExtensionApiLibraryClass::extensionApi",
"ExtensionApiLibraryInterfaceImpl::extensionApi");
@Parameter public TestParameters parameters;
@Parameters(name = "{0}")
public static TestParametersCollection data() {
return getTestParameters().withAllRuntimesAndApiLevels().build();
}
private void setupTestBuilder(TestCompilerBuilder<?, ?, ?, ?, ?> testBuilder) throws Exception {
testBuilder
.addLibraryClasses(
ExtensionApiLibraryClass.class,
ExtensionApiLibraryInterface.class,
ExtensionApiLibraryInterfaceProvider.class)
.addDefaultRuntimeLibrary(parameters)
.addInnerClasses(getClass())
.setMinApi(parameters)
.addOptionsModification(
options -> {
String typeName = ExtensionApiLibraryClass.class.getTypeName();
options.apiModelingOptions().androidApiExtensionPackages =
typeName.substring(0, typeName.lastIndexOf('.'));
})
// TODO(b/213552119): Remove when enabled by default.
.apply(ApiModelingTestHelper::enableApiCallerIdentification)
.apply(ApiModelingTestHelper::enableOutliningOfMethods)
.apply(ApiModelingTestHelper::disableStubbingOfClasses);
}
private void populateBootClasspath(TestCompileResult<?, ?> compilerResult) throws Exception {
compilerResult.addBootClasspathClasses(
ExtensionApiLibraryClass.class,
ExtensionApiLibraryInterface.class,
ExtensionApiLibraryInterfaceImpl.class,
ExtensionApiLibraryInterfaceProvider.class);
}
@Test
public void testD8Debug() throws Exception {
parameters.assumeDexRuntime();
testForD8()
.setMode(CompilationMode.DEBUG)
.apply(this::setupTestBuilder)
.compile()
.apply(this::populateBootClasspath)
.run(parameters.getRuntime(), Main.class)
.assertSuccessWithOutput(EXPECTED_OUTPUT)
.inspect(this::inspect);
}
@Test
public void testD8Release() throws Exception {
parameters.assumeDexRuntime();
testForD8()
.setMode(CompilationMode.RELEASE)
.apply(this::setupTestBuilder)
.compile()
.apply(this::populateBootClasspath)
.run(parameters.getRuntime(), Main.class)
.assertSuccessWithOutput(EXPECTED_OUTPUT)
.inspect(this::inspect);
}
@Test
public void testR8() throws Exception {
testForR8(parameters.getBackend())
.apply(this::setupTestBuilder)
.addKeepMainRule(Main.class)
.enableInliningAnnotations()
.compile()
.apply(this::populateBootClasspath)
.run(parameters.getRuntime(), Main.class)
.assertSuccessWithOutput(EXPECTED_OUTPUT)
.inspect(this::inspect);
}
private void inspect(CodeInspector inspector) throws Exception {
assertThat(inspector.clazz(ExtensionApiLibraryClass.class), isAbsent());
verifyThat(
inspector, parameters, ExtensionApiLibraryClass.class.getDeclaredMethod("extensionApi"))
.isOutlinedFromIf(
Main.class.getDeclaredMethod("main", String[].class), parameters.isDexRuntime());
verifyThat(
inspector,
parameters,
ExtensionApiLibraryInterface.class.getDeclaredMethod("extensionInterfaceApi"))
.isOutlinedFromIf(
ExtensionApiUtil.class.getDeclaredMethod("invokeExtensionInterfaceApi"),
parameters.isDexRuntime());
}
public static class ExtensionApiUtil {
private static final ExtensionApiLibraryInterface api;
static {
api = ExtensionApiLibraryInterfaceProvider.getProvider();
}
@NeverInline
public static void invokeExtensionInterfaceApi() {
api.extensionInterfaceApi();
}
}
public static class Main {
public static void main(String[] args) {
ExtensionApiLibraryClass.extensionApi();
ExtensionApiUtil.invokeExtensionInterfaceApi();
}
}
}