| // 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.kotlin.metadata; |
| |
| import static com.android.tools.r8.KotlinCompilerTool.KOTLINC; |
| import static com.android.tools.r8.utils.codeinspector.Matchers.isPresent; |
| import static com.android.tools.r8.utils.codeinspector.Matchers.isRenamed; |
| import static org.hamcrest.CoreMatchers.containsString; |
| import static org.hamcrest.CoreMatchers.not; |
| import static org.hamcrest.MatcherAssert.assertThat; |
| import static org.junit.Assert.assertNotEquals; |
| |
| import com.android.tools.r8.R8TestCompileResult; |
| import com.android.tools.r8.TestParameters; |
| import com.android.tools.r8.ToolHelper.KotlinTargetVersion; |
| import com.android.tools.r8.ToolHelper.ProcessResult; |
| import com.android.tools.r8.shaking.ProguardKeepAttributes; |
| import com.android.tools.r8.utils.InternalOptions; |
| import com.android.tools.r8.utils.codeinspector.AnnotationSubject; |
| import com.android.tools.r8.utils.codeinspector.ClassSubject; |
| import com.android.tools.r8.utils.codeinspector.MethodSubject; |
| import java.nio.file.Path; |
| import java.util.Collection; |
| import org.junit.BeforeClass; |
| import org.junit.Test; |
| import org.junit.runner.RunWith; |
| import org.junit.runners.Parameterized; |
| |
| @RunWith(Parameterized.class) |
| public class MetadataRenameInMultifileClassTest extends KotlinMetadataTestBase { |
| |
| private final TestParameters parameters; |
| |
| @Parameterized.Parameters(name = "{0} target: {1}") |
| public static Collection<Object[]> data() { |
| return buildParameters( |
| getTestParameters().withCfRuntimes().build(), KotlinTargetVersion.values()); |
| } |
| |
| public MetadataRenameInMultifileClassTest( |
| TestParameters parameters, KotlinTargetVersion targetVersion) { |
| super(targetVersion); |
| this.parameters = parameters; |
| } |
| |
| private static Path multifileLibJar; |
| |
| @BeforeClass |
| public static void createLibJar() throws Exception { |
| String multifileLibFolder = PKG_PREFIX + "/multifileclass_lib"; |
| multifileLibJar = |
| kotlinc(KOTLINC, KotlinTargetVersion.JAVA_8) |
| .addSourceFiles( |
| getKotlinFileInTest(multifileLibFolder, "signed"), |
| getKotlinFileInTest(multifileLibFolder, "unsigned")) |
| .compile(); |
| } |
| |
| @Test |
| public void testMetadataInMultifileClass_merged() throws Exception { |
| R8TestCompileResult compileResult = |
| testForR8(parameters.getBackend()) |
| .addProgramFiles(multifileLibJar) |
| // Keep UtilKt#comma*Join*(). Let R8 optimize (inline) others, such as joinOf*(String). |
| .addKeepRules("-keep class **.UtilKt") |
| .addKeepRules("-keepclassmembers class * { ** comma*Join*(...); }") |
| .addKeepAttributes(ProguardKeepAttributes.RUNTIME_VISIBLE_ANNOTATIONS) |
| .addOptionsModification(InternalOptions::enableKotlinMetadataRewriting) |
| .compile(); |
| String pkg = getClass().getPackage().getName(); |
| final String utilClassName = pkg + ".multifileclass_lib.UtilKt"; |
| final String signedClassName = pkg + ".multifileclass_lib.UtilKt__SignedKt"; |
| compileResult.inspect(inspector -> { |
| ClassSubject util = inspector.clazz(utilClassName); |
| assertThat(util, isPresent()); |
| assertThat(util, not(isRenamed())); |
| MethodSubject commaJoinOfInt = util.uniqueMethodWithName("commaSeparatedJoinOfInt"); |
| assertThat(commaJoinOfInt, isPresent()); |
| assertThat(commaJoinOfInt, not(isRenamed())); |
| MethodSubject joinOfInt = util.uniqueMethodWithName("joinOfInt"); |
| assertThat(joinOfInt, not(isPresent())); |
| // API entry is kept, hence the presence of Metadata. |
| AnnotationSubject annotationSubject = util.annotation(METADATA_TYPE); |
| assertThat(annotationSubject, isPresent()); |
| // TODO(b/70169921): need further inspection. |
| |
| ClassSubject signed = inspector.clazz(signedClassName); |
| assertThat(signed, isPresent()); |
| assertThat(signed, isRenamed()); |
| commaJoinOfInt = signed.uniqueMethodWithName("commaSeparatedJoinOfInt"); |
| assertThat(commaJoinOfInt, isPresent()); |
| assertThat(commaJoinOfInt, not(isRenamed())); |
| joinOfInt = signed.uniqueMethodWithName("joinOfInt"); |
| assertThat(joinOfInt, isPresent()); |
| assertThat(joinOfInt, isRenamed()); |
| // API entry is kept, hence the presence of Metadata. |
| annotationSubject = util.annotation(METADATA_TYPE); |
| assertThat(annotationSubject, isPresent()); |
| // TODO(b/70169921): need further inspection. |
| }); |
| |
| Path libJar = compileResult.writeToZip(); |
| |
| String appFolder = PKG_PREFIX + "/multifileclass_app"; |
| ProcessResult kotlinTestCompileResult = |
| kotlinc(parameters.getRuntime().asCf(), KOTLINC, KotlinTargetVersion.JAVA_8) |
| .addClasspathFiles(libJar) |
| .addSourceFiles(getKotlinFileInTest(appFolder, "main")) |
| .setOutputPath(temp.newFolder().toPath()) |
| // TODO(b/143687784): update to just .compile() once fixed. |
| .compileRaw(); |
| // TODO(b/70169921): should be able to compile! |
| assertNotEquals(0, kotlinTestCompileResult.exitCode); |
| assertThat(kotlinTestCompileResult.stderr, containsString("unresolved reference: join")); |
| } |
| |
| @Test |
| public void testMetadataInMultifileClass_renamed() throws Exception { |
| R8TestCompileResult compileResult = |
| testForR8(parameters.getBackend()) |
| .addProgramFiles(multifileLibJar) |
| // Keep UtilKt#comma*Join*(). |
| .addKeepRules("-keep class **.UtilKt") |
| .addKeepRules("-keepclassmembers class * { ** comma*Join*(...); }") |
| // Keep yet rename joinOf*(String). |
| .addKeepRules( |
| "-keepclassmembers,allowobfuscation class * { ** joinOf*(...); }") |
| .addKeepAttributes(ProguardKeepAttributes.RUNTIME_VISIBLE_ANNOTATIONS) |
| .addOptionsModification(InternalOptions::enableKotlinMetadataRewriting) |
| .compile(); |
| String pkg = getClass().getPackage().getName(); |
| final String utilClassName = pkg + ".multifileclass_lib.UtilKt"; |
| final String signedClassName = pkg + ".multifileclass_lib.UtilKt__SignedKt"; |
| compileResult.inspect(inspector -> { |
| ClassSubject util = inspector.clazz(utilClassName); |
| assertThat(util, isPresent()); |
| assertThat(util, not(isRenamed())); |
| MethodSubject commaJoinOfInt = util.uniqueMethodWithName("commaSeparatedJoinOfInt"); |
| assertThat(commaJoinOfInt, isPresent()); |
| assertThat(commaJoinOfInt, not(isRenamed())); |
| MethodSubject joinOfInt = util.uniqueMethodWithName("joinOfInt"); |
| assertThat(joinOfInt, isPresent()); |
| assertThat(joinOfInt, isRenamed()); |
| // API entry is kept, hence the presence of Metadata. |
| AnnotationSubject annotationSubject = util.annotation(METADATA_TYPE); |
| assertThat(annotationSubject, isPresent()); |
| // TODO(b/70169921): need further inspection. |
| |
| ClassSubject signed = inspector.clazz(signedClassName); |
| assertThat(signed, isPresent()); |
| assertThat(signed, isRenamed()); |
| commaJoinOfInt = signed.uniqueMethodWithName("commaSeparatedJoinOfInt"); |
| assertThat(commaJoinOfInt, isPresent()); |
| assertThat(commaJoinOfInt, not(isRenamed())); |
| joinOfInt = signed.uniqueMethodWithName("joinOfInt"); |
| assertThat(joinOfInt, isPresent()); |
| assertThat(joinOfInt, isRenamed()); |
| // API entry is kept, hence the presence of Metadata. |
| annotationSubject = util.annotation(METADATA_TYPE); |
| assertThat(annotationSubject, isPresent()); |
| // TODO(b/70169921): need further inspection. |
| }); |
| |
| Path libJar = compileResult.writeToZip(); |
| |
| String appFolder = PKG_PREFIX + "/multifileclass_app"; |
| ProcessResult kotlinTestCompileResult = |
| kotlinc(parameters.getRuntime().asCf(), KOTLINC, KotlinTargetVersion.JAVA_8) |
| .addClasspathFiles(libJar) |
| .addSourceFiles(getKotlinFileInTest(appFolder, "main")) |
| .setOutputPath(temp.newFolder().toPath()) |
| // TODO(b/143687784): update to just .compile() once fixed. |
| .compileRaw(); |
| // TODO(b/70169921): should be able to compile! |
| assertNotEquals(0, kotlinTestCompileResult.exitCode); |
| assertThat(kotlinTestCompileResult.stderr, containsString("unresolved reference: join")); |
| } |
| } |