blob: fc1f76d78a57d5416d0c2244006e8dec09c3119c [file] [log] [blame]
// Copyright (c) 2025, 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.androidresources;
import com.android.tools.r8.R8TestBuilder;
import com.android.tools.r8.TestBase;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.ToolHelper;
import com.android.tools.r8.androidresources.AndroidResourceTestingUtils.AndroidTestResource;
import com.android.tools.r8.androidresources.AndroidResourceTestingUtils.AndroidTestResourceBuilder;
import com.android.tools.r8.androidresources.ResourceShrinkingWithSeveralFeaturesTest.FeatureSplit.FeatureSplitMain;
import com.android.tools.r8.androidresources.ResourceShrinkingWithSeveralFeaturesTest.FeatureSplit2.FeatureSplit2Main;
import com.android.tools.r8.utils.BooleanUtils;
import com.google.common.collect.ImmutableList;
import java.io.IOException;
import java.util.List;
import org.junit.Assume;
import org.junit.Test;
import org.junit.rules.TemporaryFolder;
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 ResourceShrinkingWithSeveralFeaturesTest extends TestBase {
@Parameter(0)
public TestParameters parameters;
@Parameter(1)
public boolean optimized;
@Parameters(name = "{0}, optimized: {1}")
public static List<Object[]> data() {
return buildParameters(
getTestParameters()
.withDefaultDexRuntime()
.withAllApiLevels()
.withPartialCompilation()
.build(),
BooleanUtils.values());
}
public static AndroidTestResource getTestResources(TemporaryFolder temp) throws Exception {
return new AndroidTestResourceBuilder()
.withSimpleManifestAndAppNameString()
.addRClassInitializeWithDefaultValues(R.string.class)
.build(temp);
}
public AndroidTestResource getFeatureSplitTestResources(TemporaryFolder temp) throws IOException {
return new AndroidTestResourceBuilder()
.withSimpleManifestAndAppNameString()
.setPackageId(0x80)
.addRClassInitializeWithDefaultValues(FeatureSplit.R.string.class)
.build(temp);
}
public AndroidTestResource getFeatureSplit2TestResources(TemporaryFolder temp)
throws IOException {
return new AndroidTestResourceBuilder()
.withSimpleManifestAndAppNameString()
.setPackageId(0x81)
.addRClassInitializeWithDefaultValues(FeatureSplit2.R.string.class)
.build(temp);
}
@Test
public void testR8() throws Exception {
Assume.assumeTrue(optimized || parameters.getPartialCompilationTestParameters().isNone());
TemporaryFolder featureSplitTemp = ToolHelper.getTemporaryFolderForTest();
featureSplitTemp.create();
TemporaryFolder featureSplit2Temp = ToolHelper.getTemporaryFolderForTest();
featureSplit2Temp.create();
testForR8(parameters)
.addProgramClasses(Base.class)
.addFeatureSplit(FeatureSplitMain.class)
.addFeatureSplit(FeatureSplit2Main.class)
.addAndroidResources(getTestResources(temp))
.addFeatureSplitAndroidResources(
getFeatureSplitTestResources(featureSplitTemp), FeatureSplit.class.getName())
.addFeatureSplitAndroidResources(
getFeatureSplit2TestResources(featureSplit2Temp), FeatureSplit2.class.getName())
.applyIf(optimized, R8TestBuilder::enableOptimizedShrinking)
.addResourceShrinkerLogCapture()
.addKeepMainRule(Base.class)
.addKeepMainRule(FeatureSplitMain.class)
.addKeepMainRule(FeatureSplit2Main.class)
.compile()
.inspectShrunkenResources(
resourceTableInspector -> {
resourceTableInspector.assertContainsResourceWithName("string", "base_used");
if (!parameters.isRandomPartialCompilation()) {
resourceTableInspector.assertDoesNotContainResourceWithName(
"string", "base_unused");
}
})
.inspectShrunkenResourcesForFeature(
resourceTableInspector -> {
resourceTableInspector.assertContainsResourceWithName("string", "feature_used");
if (!parameters.isRandomPartialCompilation()) {
resourceTableInspector.assertDoesNotContainResourceWithName(
"string", "feature_unused");
}
},
FeatureSplit.class.getName())
.inspectShrunkenResourcesForFeature(
resourceTableInspector -> {
resourceTableInspector.assertContainsResourceWithName("string", "feature2_used");
if (!parameters.isRandomPartialCompilation()) {
resourceTableInspector.assertDoesNotContainResourceWithName(
"string", "feature2_unused");
}
},
FeatureSplit2.class.getName())
.inspectResourceShrinkerLog(
inspector -> {
for (String dexReachableString :
ImmutableList.of("base_used", "feature_used", "feature2_used")) {
if (optimized) {
inspector.ensureReachableOptimized("string", dexReachableString);
} else {
inspector.ensureDexReachable("string", dexReachableString);
}
}
for (String unused :
ImmutableList.of("base_unused", "feature_unused", "feature2_unused")) {
if (optimized) {
if (!parameters.isRandomPartialCompilation()) {
inspector.ensureUnreachableOptimized("string", unused);
}
} else {
inspector.ensureUnreachable("string", unused);
}
}
})
.run(parameters.getRuntime(), Base.class)
.assertSuccess();
}
public static class Base {
public static void main(String[] args) {
if (System.currentTimeMillis() == 0) {
System.out.println(R.string.base_used);
}
}
}
public static class R {
public static class string {
public static int base_used;
public static int base_unused;
}
}
public static class FeatureSplit {
public static class FeatureSplitMain {
public static void main(String[] args) {
if (System.currentTimeMillis() == 0) {
System.out.println(R.string.feature_used);
}
}
}
public static class R {
public static class string {
public static int feature_used;
public static int feature_unused;
}
}
}
public static class FeatureSplit2 {
public static class FeatureSplit2Main {
public static void main(String[] args) {
if (System.currentTimeMillis() == 0) {
System.out.println(R.string.feature2_used);
}
}
}
public static class R {
public static class string {
public static int feature2_used;
public static int feature2_unused;
}
}
}
}