blob: 25cbe9a374387757230b175bd7c59de06f8f8169 [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.dexsplitter;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotEquals;
import static org.junit.Assert.assertTrue;
import static org.junit.Assume.assumeTrue;
import com.android.tools.r8.CompilationFailedException;
import com.android.tools.r8.FeatureSplit;
import com.android.tools.r8.NeverInline;
import com.android.tools.r8.NeverMerge;
import com.android.tools.r8.R8FullTestBuilder;
import com.android.tools.r8.R8TestCompileResult;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.TestParametersCollection;
import com.android.tools.r8.ToolHelper.ProcessResult;
import com.android.tools.r8.utils.StringUtils;
import com.android.tools.r8.utils.codeinspector.ClassSubject;
import com.google.common.collect.ImmutableSet;
import java.io.IOException;
import java.util.concurrent.ExecutionException;
import java.util.function.Consumer;
import java.util.function.Predicate;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameters;
@RunWith(Parameterized.class)
public class DexSplitterMergeRegression extends SplitterTestBase {
public static final String EXPECTED = StringUtils.lines("42", "foobar");
@Parameters(name = "{0}")
public static TestParametersCollection params() {
return getTestParameters().withAllRuntimes().build();
}
private final TestParameters parameters;
public DexSplitterMergeRegression(TestParameters parameters) {
this.parameters = parameters;
}
@Test
public void testInliningFromFeature() throws Exception {
// Static merging is based on sorting order, we assert that we merged to the feature.
Predicate<R8TestCompileResult> ensureMergingToFeature =
r8TestCompileResult -> {
try {
ClassSubject clazz = r8TestCompileResult.inspector().clazz(AFeatureWithStatic.class);
return clazz.allMethods().size() == 2
&& clazz.uniqueMethodWithName("getBase42").isPresent()
&& clazz.uniqueMethodWithName("getFoobar").isPresent();
} catch (IOException | ExecutionException ex) {
throw new RuntimeException("Failed lookup up AFeatureWithStatic");
}
};
Consumer<R8FullTestBuilder> configurator =
r8FullTestBuilder ->
r8FullTestBuilder
.enableMergeAnnotations()
.enableInliningAnnotations()
.noMinification()
.addOptionsModification(o -> o.testing.deterministicSortingBasedOnDexType = true);
ProcessResult processResult =
testDexSplitter(
parameters,
ImmutableSet.of(BaseClass.class, BaseWithStatic.class),
ImmutableSet.of(FeatureClass.class, AFeatureWithStatic.class),
FeatureClass.class,
EXPECTED,
ensureMergingToFeature,
configurator);
// We expect art to fail on this with the dex splitter.
assertNotEquals(processResult.exitCode, 0);
assertTrue(processResult.stderr.contains("NoClassDefFoundError"));
}
@Test
public void testOnR8Splitter() throws IOException, CompilationFailedException,
ExecutionException {
assumeTrue(parameters.isDexRuntime());
Consumer<R8FullTestBuilder> configurator =
r8FullTestBuilder -> r8FullTestBuilder.enableMergeAnnotations().noMinification();
ProcessResult processResult =
testR8Splitter(
parameters,
ImmutableSet.of(BaseClass.class, BaseWithStatic.class),
ImmutableSet.of(FeatureClass.class, AFeatureWithStatic.class),
FeatureClass.class,
EXPECTED,
a -> true,
configurator);
assertEquals(processResult.exitCode, 0);
assertTrue(processResult.stdout.equals(StringUtils.lines("42", "foobar")));
}
@NeverMerge
public static class BaseClass implements RunInterface {
@NeverInline
public void run() {
System.out.println(BaseWithStatic.getBase42());
}
}
public static class BaseWithStatic {
@NeverInline
public static int getBase42() {
if (System.currentTimeMillis() < 2) {
return 43;
} else {
return 42;
}
}
}
public static class FeatureClass extends BaseClass {
public void run() {
super.run();
System.out.println(AFeatureWithStatic.getFoobar());
}
}
// Name is important, see predicate in tests/
public static class AFeatureWithStatic {
@NeverInline
public static String getFoobar() {
if (System.currentTimeMillis() < 2) {
return "barFoo";
} else {
return "foobar";
}
}
}
}