blob: b0557d3ebc58f26875bfbb27ed527bf6e44e9eb3 [file] [log] [blame]
// Copyright (c) 2018, 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.classmerging;
import static com.android.tools.r8.utils.codeinspector.Matchers.isPresent;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.core.IsNot.not;
import com.android.tools.r8.NeverClassInline;
import com.android.tools.r8.NeverInline;
import com.android.tools.r8.TestBase;
import com.android.tools.r8.ToolHelper;
import com.android.tools.r8.ToolHelper.DexVm.Version;
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.Parameters;
@RunWith(Parameterized.class)
public class StaticClassMergerInterfaceTest extends TestBase {
private final Backend backend;
@Parameters(name = "Backend: {0}")
public static Backend[] data() {
return ToolHelper.getBackends();
}
public StaticClassMergerInterfaceTest(Backend backend) {
this.backend = backend;
}
@Test
public void test() throws Exception {
String expectedOutput = StringUtils.lines("In A.a()", "In B.b()", "In C.c()");
CodeInspector inspector =
testForR8(backend)
.addInnerClasses(StaticClassMergerInterfaceTest.class)
.addKeepMainRule(TestClass.class)
.addKeepRules("-dontobfuscate")
.enableInliningAnnotations()
.enableNeverClassInliningAnnotations()
.run(TestClass.class)
.assertSuccessWithOutput(expectedOutput)
.inspector();
// Check that A has not been merged into B. The static class merger visits classes in alpha-
// betical order. By the time A is processed, there is no merge representative and A is not
// a valid merge representative itself, because it is an interface.
if (ToolHelper.getDexVm().getVersion().isNewerThan(Version.V6_0_1) || backend == Backend.CF) {
assertThat(inspector.clazz(A.class), isPresent());
} else {
assertThat(inspector.clazz(A.class.getTypeName() + "$-CC"), isPresent());
}
// By the time B is processed, there is no merge representative, so it should be present.
assertThat(inspector.clazz(B.class), isPresent());
// By the time C is processed, B should be merge candidate. Therefore, we should allow C.c() to
// be moved to B *although C is an interface*.
assertThat(inspector.clazz(C.class), not(isPresent()));
}
static class TestClass {
public static void main(String[] args) {
A.a();
B.b();
C.c();
}
}
@NeverClassInline
interface A {
@NeverInline
static void a() {
System.out.println("In A.a()");
}
}
@NeverClassInline
static class B {
@NeverInline
static void b() {
System.out.println("In B.b()");
}
}
@NeverClassInline
interface C {
@NeverInline
static void c() {
System.out.println("In C.c()");
}
}
}