blob: 446978fcef91ff7e7b24a0e3e2311dd46d180cad [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.graph.genericsignature;
import static org.hamcrest.CoreMatchers.containsString;
import static org.junit.Assert.assertFalse;
import static org.junit.Assume.assumeTrue;
import com.android.tools.r8.TestBase;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.TestParametersCollection;
import com.android.tools.r8.ToolHelper;
import com.android.tools.r8.graph.genericsignature.GenericSignatureSelfReferencingInnerClassesTest.A.B;
import com.android.tools.r8.references.Reference;
import com.android.tools.r8.utils.StringUtils;
import java.io.IOException;
import java.nio.file.Path;
import java.util.Iterator;
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 GenericSignatureSelfReferencingInnerClassesTest extends TestBase {
@Parameter(0)
public TestParameters parameters;
@Parameters(name = "{0}")
public static TestParametersCollection data() {
return getTestParameters().withAllRuntimesAndApiLevels().build();
}
private static final String CLASS_B = "class " + A.B.class.getTypeName();
private static final String EXPECTED_OUTPUT = StringUtils.lines(CLASS_B, CLASS_B);
@Test
public void testJvm() throws Exception {
parameters.assumeJvmTestParameters();
testForJvm(parameters)
.addProgramClasses(TestClass.class)
.addProgramClassFileData(getProgramClassFileDataWithSelfReferencingInnerClass())
.run(parameters.getRuntime(), TestClass.class)
.assertFailureWithErrorThatThrows(ClassFormatError.class);
}
@Test
public void testD8() throws Exception {
assumeTrue(parameters.isDexRuntime());
testForD8(parameters.getBackend())
.addProgramClasses(TestClass.class)
.addProgramClassFileData(getProgramClassFileDataWithSelfReferencingInnerClass())
.setMinApi(parameters.getApiLevel())
.run(parameters.getRuntime(), TestClass.class)
.assertSuccessWithOutput(EXPECTED_OUTPUT);
}
@Test
public void testR8() throws Exception {
testForR8(parameters.getBackend())
.addProgramClasses(TestClass.class)
.addProgramClassFileData(getProgramClassFileDataWithSelfReferencingInnerClass())
.addKeepMainRule(TestClass.class)
.addKeepAttributeInnerClassesAndEnclosingMethod()
.addKeepRules("-keep class **B { <methods>; }")
.setMinApi(parameters)
.allowDiagnosticInfoMessages()
.compile()
.assertAllInfoMessagesMatch(containsString("Malformed inner-class attribute"))
.run(parameters.getRuntime(), TestClass.class)
.applyIf(
parameters.isCfRuntime(),
r -> r.assertFailureWithErrorThatThrows(ClassFormatError.class),
r -> r.assertSuccessWithOutput(EXPECTED_OUTPUT));
}
@Test
public void testR8NoKeep() throws Exception {
testForR8(parameters.getBackend())
.addProgramClasses(TestClass.class)
.addProgramClassFileData(getProgramClassFileDataWithSelfReferencingInnerClass())
.addKeepMainRule(TestClass.class)
.setMinApi(parameters)
.allowDiagnosticInfoMessages()
.compile()
.assertAllInfoMessagesMatch(containsString("Malformed inner-class attribute"))
.run(parameters.getRuntime(), TestClass.class)
.assertSuccessWithOutputLines("class " + getClass().getPackage().getName() + ".a", "null");
}
// Change the InnerClasses attribute for the inner class B to be B itself.
private byte[] getProgramClassFileDataWithSelfReferencingInnerClass() throws IOException {
Iterator<Path> innerClassesIterator =
ToolHelper.getClassFilesForInnerClasses(A.class).iterator();
Path innerClass = innerClassesIterator.next();
assertFalse(innerClassesIterator.hasNext());
return transformer(innerClass, Reference.classFromBinaryName(binaryName(A.B.class)))
.rewriteInnerClass(binaryName(A.B.class), binaryName(A.B.class), "B")
.transform();
}
static class A {
static class B {}
}
static class TestClass {
public static void main(String[] args) {
System.out.println(B.class);
System.out.println(B.class.getEnclosingClass());
}
}
}