blob: d53e0e69afee5e8bbc96920ad3aa14c69dac7467 [file] [log] [blame]
// Copyright (c) 2020, 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.desugar.desugaredlibrary;
import static com.android.tools.r8.MarkerMatcher.assertMarkersMatch;
import static com.android.tools.r8.MarkerMatcher.markerCompilationMode;
import static com.android.tools.r8.MarkerMatcher.markerHasDesugaredLibraryIdentifier;
import static com.android.tools.r8.MarkerMatcher.markerIsDesugared;
import static com.android.tools.r8.MarkerMatcher.markerMinApi;
import static com.android.tools.r8.MarkerMatcher.markerTool;
import static org.hamcrest.CoreMatchers.allOf;
import static org.hamcrest.CoreMatchers.not;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertFalse;
import static org.junit.Assert.assertTrue;
import com.android.tools.r8.CompilationFailedException;
import com.android.tools.r8.CompilationMode;
import com.android.tools.r8.D8TestCompileResult;
import com.android.tools.r8.Diagnostic;
import com.android.tools.r8.ExtractMarker;
import com.android.tools.r8.TestDiagnosticMessages;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.TestParametersCollection;
import com.android.tools.r8.ToolHelper;
import com.android.tools.r8.desugar.desugaredlibrary.jdktests.Jdk11DesugaredLibraryTestBase;
import com.android.tools.r8.dex.Marker;
import com.android.tools.r8.dex.Marker.Tool;
import com.android.tools.r8.utils.AndroidApiLevel;
import com.google.common.collect.ImmutableList;
import java.nio.file.Path;
import java.util.ArrayList;
import java.util.List;
import org.hamcrest.Matcher;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
@RunWith(Parameterized.class)
public class MergingWithDesugaredLibraryTest extends Jdk11DesugaredLibraryTestBase {
private static final String JAVA_RESULT = "java.util.stream.ReferencePipeline$Head";
private static final String J$_RESULT = "j$.util.stream.ReferencePipeline$Head";
private final TestParameters parameters;
@Parameterized.Parameters(name = "{0}")
public static TestParametersCollection data() {
return getTestParameters().withDexRuntimes().withAllApiLevels().build();
}
public MergingWithDesugaredLibraryTest(TestParameters parameters) {
this.parameters = parameters;
}
@Test
public void testMergeDesugaredAndNonDesugared() throws Exception {
D8TestCompileResult compileResult;
try {
compileResult =
testForD8()
.addProgramFiles(buildPart1DesugaredLibrary(), buildPart2NoDesugaredLibrary())
.addLibraryFiles(ToolHelper.getAndroidJar(AndroidApiLevel.P))
.setMinApi(parameters.getApiLevel())
.enableCoreLibraryDesugaring(parameters.getApiLevel())
.compileWithExpectedDiagnostics(this::assertError);
assertFalse(expectError());
} catch (CompilationFailedException e) {
assertTrue(expectError());
return;
}
assert !expectError();
assert compileResult != null;
compileResult.addDesugaredCoreLibraryRunClassPath(
this::buildDesugaredLibrary, parameters.getApiLevel());
compileResult
.run(parameters.getRuntime(), Part1.class)
.assertSuccessWithOutputLines(JAVA_RESULT);
compileResult
.run(parameters.getRuntime(), Part2.class)
.assertSuccessWithOutputLines(JAVA_RESULT);
}
@Test
public void testMergeDesugaredWithShrunkenLib() throws Exception {
// Compile a library with R8 to CF.
Path shrunkenLib =
testForR8(Backend.CF)
.addProgramClasses(Part2.class)
.addKeepClassRules(Part2.class)
.compile()
.writeToZip();
// R8 class file output marker has no library desugaring identifier.
Matcher<Marker> libraryMatcher =
allOf(
markerTool(Tool.R8),
markerCompilationMode(CompilationMode.RELEASE),
not(markerIsDesugared()),
not(markerHasDesugaredLibraryIdentifier()));
assertMarkersMatch(ExtractMarker.extractMarkerFromJarFile(shrunkenLib), libraryMatcher);
// Build an app with the R8 compiled library.
Path app =
testForD8()
.addProgramFiles(buildPart1DesugaredLibrary(), shrunkenLib)
.addLibraryFiles(ToolHelper.getAndroidJar(AndroidApiLevel.P))
.setMinApi(parameters.getApiLevel())
.enableCoreLibraryDesugaring(parameters.getApiLevel())
.compile()
.writeToZip();
// The app has both the R8 marker from the library compilation and the D8 marker from dexing.
Matcher<Marker> d8Matcher =
allOf(
markerTool(Tool.D8),
markerIsDesugared(),
markerHasDesugaredLibraryIdentifier(
parameters.getApiLevel().isLessThan(AndroidApiLevel.O)));
assertMarkersMatch(
ExtractMarker.extractMarkerFromDexFile(app), ImmutableList.of(libraryMatcher, d8Matcher));
}
@Test
public void testMergeDesugaredWithDesugaredLib() throws Exception {
// Compile a library with D8 to CF.
Path desugaredLibCf =
testForD8(Backend.CF)
.addProgramClasses(Part2.class)
.setMinApi(parameters.getApiLevel())
.compile()
.writeToZip();
// D8 class file output marker has desugaring but no library desugaring.
Matcher<Marker> markerMatcher =
allOf(
markerTool(Tool.D8),
markerCompilationMode(CompilationMode.DEBUG),
markerIsDesugared(),
markerMinApi(parameters.getApiLevel()),
not(markerHasDesugaredLibraryIdentifier()));
assertMarkersMatch(ExtractMarker.extractMarkerFromJarFile(desugaredLibCf), markerMatcher);
Path desugaredLibDex =
testForD8()
.addProgramFiles(desugaredLibCf)
.disableDesugaring()
.setMinApi(parameters.getApiLevel())
.compile()
.writeToZip();
// D8 dex file output marker has the same marker as the D8 class file output.
assertMarkersMatch(ExtractMarker.extractMarkerFromJarFile(desugaredLibDex), markerMatcher);
// Build an app using library desugaring merging with library not using library desugaring.
Path app;
try {
app =
testForD8()
.addProgramFiles(buildPart1DesugaredLibrary(), desugaredLibDex)
.setMinApi(parameters.getApiLevel())
.compile()
.writeToZip();
assertMarkersMatch(ExtractMarker.extractMarkerFromDexFile(app), markerMatcher);
} catch (CompilationFailedException e) {
assertTrue(someLibraryDesugaringRequired());
return;
}
assert !someLibraryDesugaringRequired();
}
private void assertError(TestDiagnosticMessages m) {
List<Diagnostic> errors = m.getErrors();
if (expectError()) {
assertEquals(1, errors.size());
assertTrue(
errors.stream()
.anyMatch(
w ->
w.getDiagnosticMessage()
.contains(
"The compilation is merging inputs with different"
+ " desugared library desugaring")));
} else {
assertEquals(0, errors.size());
}
}
private boolean expectError() {
return someLibraryDesugaringRequired();
}
private boolean someLibraryDesugaringRequired() {
return parameters.getApiLevel().getLevel() <= AndroidApiLevel.N.getLevel();
}
@Test
public void testMergeDesugaredAndClassFile() throws Exception {
D8TestCompileResult compileResult =
testForD8()
.addProgramFiles(buildPart1DesugaredLibrary())
.addProgramClasses(Part2.class)
.addLibraryFiles(ToolHelper.getAndroidJar(AndroidApiLevel.P))
.setMinApi(parameters.getApiLevel())
.enableCoreLibraryDesugaring(parameters.getApiLevel())
.compile()
.inspectDiagnosticMessages(this::assertWarningPresent)
.addDesugaredCoreLibraryRunClassPath(
this::buildDesugaredLibrary, parameters.getApiLevel());
if (parameters.getApiLevel().getLevel() < AndroidApiLevel.N.getLevel()) {
compileResult
.run(parameters.getRuntime(), Part1.class)
.assertSuccessWithOutputLines(J$_RESULT);
compileResult
.run(parameters.getRuntime(), Part2.class)
.assertSuccessWithOutputLines(J$_RESULT);
} else {
compileResult
.run(parameters.getRuntime(), Part1.class)
.assertSuccessWithOutputLines(JAVA_RESULT);
compileResult
.run(parameters.getRuntime(), Part2.class)
.assertSuccessWithOutputLines(JAVA_RESULT);
}
}
private void assertWarningPresent(TestDiagnosticMessages testDiagnosticMessages) {
if (parameters.getApiLevel().getLevel() > AndroidApiLevel.N.getLevel()) {
return;
}
assertTrue(
testDiagnosticMessages.getWarnings().stream()
.anyMatch(
warn -> warn.getDiagnosticMessage().startsWith("The compilation is slowed down")));
}
private Path buildPart1DesugaredLibrary() throws Exception {
return testForD8()
.addProgramClasses(Part1.class)
.setMinApi(parameters.getApiLevel())
.enableCoreLibraryDesugaring(parameters.getApiLevel())
.compile()
.writeToZip();
}
private Path buildPart2NoDesugaredLibrary() throws Exception {
return testForD8()
.addProgramClasses(Part2.class)
.setMinApi(parameters.getApiLevel())
.compile()
.writeToZip();
}
@SuppressWarnings("RedundantOperationOnEmptyContainer")
static class Part1 {
public static void main(String[] args) {
System.out.println(new ArrayList<>().stream().getClass().getName());
}
}
@SuppressWarnings("RedundantOperationOnEmptyContainer")
static class Part2 {
public static void main(String[] args) {
System.out.println(new ArrayList<>().stream().getClass().getName());
}
}
}