|  | // 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.repackage; | 
|  |  | 
|  | import static com.android.tools.r8.shaking.ProguardConfigurationParser.FLATTEN_PACKAGE_HIERARCHY; | 
|  | import static com.android.tools.r8.shaking.ProguardConfigurationParser.REPACKAGE_CLASSES; | 
|  | import static com.android.tools.r8.utils.codeinspector.Matchers.isPresentAndNotRenamed; | 
|  | import static org.hamcrest.MatcherAssert.assertThat; | 
|  |  | 
|  | import com.android.tools.r8.NeverInline; | 
|  | import com.android.tools.r8.TestParameters; | 
|  | import com.android.tools.r8.TestRuntime.CfVm; | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import java.util.List; | 
|  | 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 RepackageWithNestAndKeepOnPrivateTest extends RepackageTestBase { | 
|  |  | 
|  | private final String EXPECTED = "Hello World!"; | 
|  |  | 
|  | @Parameters(name = "{1}, kind: {0}") | 
|  | public static List<Object[]> data() { | 
|  | return buildParameters( | 
|  | ImmutableList.of(FLATTEN_PACKAGE_HIERARCHY, REPACKAGE_CLASSES), | 
|  | getTestParameters().withCfRuntimesStartingFromIncluding(CfVm.JDK11).build()); | 
|  | } | 
|  |  | 
|  | public RepackageWithNestAndKeepOnPrivateTest( | 
|  | String flattenPackageHierarchyOrRepackageClasses, TestParameters parameters) { | 
|  | super(flattenPackageHierarchyOrRepackageClasses, parameters); | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void testJvm() throws Exception { | 
|  | testForRuntime(parameters) | 
|  | .addProgramClassFileData(getProgramClassData()) | 
|  | .run(parameters.getRuntime(), Main.class) | 
|  | .assertSuccessWithOutputLines(EXPECTED); | 
|  | } | 
|  |  | 
|  | @Test | 
|  | public void testR8() throws Exception { | 
|  | testForR8(parameters.getBackend()) | 
|  | .addProgramClassFileData(getProgramClassData()) | 
|  | .apply(this::configureRepackaging) | 
|  | .addKeepClassAndMembersRules(A.class) | 
|  | .enableInliningAnnotations() | 
|  | .addKeepMainRule(Main.class) | 
|  | .compile() | 
|  | .inspect( | 
|  | inspector -> { | 
|  | assertThat(inspector.clazz(A.class), isPresentAndNotRenamed()); | 
|  | assertThat(B.class, isNotRepackaged(inspector)); | 
|  | }) | 
|  | .run(parameters.getRuntime(), Main.class) | 
|  | .assertSuccessWithOutputLines(EXPECTED); | 
|  | } | 
|  |  | 
|  | private List<byte[]> getProgramClassData() throws Exception { | 
|  | return ImmutableList.of( | 
|  | transformer(Main.class).removeInnerClasses().transform(), | 
|  | transformer(A.class) | 
|  | .setPrivate(A.class.getDeclaredMethod("aPrivate")) | 
|  | .removeInnerClasses() | 
|  | .setNest(A.class, B.class) | 
|  | .transform(), | 
|  | transformer(B.class).removeInnerClasses().setNest(A.class, B.class).transform()); | 
|  | } | 
|  |  | 
|  | public static class A { | 
|  |  | 
|  | @NeverInline | 
|  | /* private */ static void aPrivate() { | 
|  | System.out.println("Hello World!"); | 
|  | } | 
|  | } | 
|  |  | 
|  | public static class B { | 
|  |  | 
|  | @NeverInline | 
|  | public static void foo() { | 
|  | A.aPrivate(); | 
|  | } | 
|  | } | 
|  |  | 
|  | public static class Main { | 
|  |  | 
|  | public static void main(String[] args) { | 
|  | B.foo(); | 
|  | } | 
|  | } | 
|  | } |