Add warning and diagnostic for final fields in R classes
This will only run if we are in optimized resource shrinking
Bug: 335772436
Change-Id: Id7fe5a830625a996b97e3d490cfd439a1044e1de
diff --git a/src/main/java/com/android/tools/r8/errors/FinalRClassEntriesWithOptimizedShrinkingDiagnostic.java b/src/main/java/com/android/tools/r8/errors/FinalRClassEntriesWithOptimizedShrinkingDiagnostic.java
new file mode 100644
index 0000000..c2f18a5
--- /dev/null
+++ b/src/main/java/com/android/tools/r8/errors/FinalRClassEntriesWithOptimizedShrinkingDiagnostic.java
@@ -0,0 +1,41 @@
+// 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.errors;
+
+import com.android.tools.r8.Diagnostic;
+import com.android.tools.r8.graph.DexField;
+import com.android.tools.r8.keepanno.annotations.KeepForApi;
+import com.android.tools.r8.origin.Origin;
+import com.android.tools.r8.position.Position;
+
+@KeepForApi
+public class FinalRClassEntriesWithOptimizedShrinkingDiagnostic implements Diagnostic {
+
+ private final Origin origin;
+ private final DexField dexField;
+
+ public FinalRClassEntriesWithOptimizedShrinkingDiagnostic(Origin origin, DexField dexField) {
+ this.origin = origin;
+ this.dexField = dexField;
+ }
+
+ @Override
+ public Origin getOrigin() {
+ return origin;
+ }
+
+ @Override
+ public Position getPosition() {
+ return Position.UNKNOWN;
+ }
+
+ @Override
+ public String getDiagnosticMessage() {
+ return "Running optimized resource shrinking with final R class ids is not supported and "
+ + "can lead to missing resources and code necessary for program execution. "
+ + "Field "
+ + dexField
+ + " is final";
+ }
+}
diff --git a/src/main/java/com/android/tools/r8/graph/analysis/ResourceAccessAnalysis.java b/src/main/java/com/android/tools/r8/graph/analysis/ResourceAccessAnalysis.java
index 01e5f1e..f70748d 100644
--- a/src/main/java/com/android/tools/r8/graph/analysis/ResourceAccessAnalysis.java
+++ b/src/main/java/com/android/tools/r8/graph/analysis/ResourceAccessAnalysis.java
@@ -5,6 +5,7 @@
package com.android.tools.r8.graph.analysis;
import com.android.build.shrinker.r8integration.R8ResourceShrinkerState;
+import com.android.tools.r8.errors.FinalRClassEntriesWithOptimizedShrinkingDiagnostic;
import com.android.tools.r8.errors.Unreachable;
import com.android.tools.r8.graph.AppInfoWithClassHierarchy;
import com.android.tools.r8.graph.AppView;
@@ -90,7 +91,6 @@
}
}
- @SuppressWarnings("ReferenceEquality")
private void populateRClassValues(ProgramField field) {
// TODO(287398085): Pending discussions with the AAPT2 team, we might need to harden this
// to not fail if we wrongly classify an unrelated class as R class in our heuristic..
@@ -100,10 +100,28 @@
if (programClassInitializer != null) {
analyzeClassInitializer(rClassValueBuilder, programClassInitializer);
}
-
+ warnOnFinalIdFields(field.getHolder());
fieldToValueMapping.put(field.getHolderType(), rClassValueBuilder.build());
}
+ private void warnOnFinalIdFields(DexProgramClass holder) {
+ if (!appView.options().isOptimizedResourceShrinking()) {
+ return;
+ }
+ for (DexEncodedField field : holder.fields()) {
+ if (field.isStatic()
+ && field.isFinal()
+ && field.hasExplicitStaticValue()
+ && field.getType().isIntType()) {
+ appView
+ .reporter()
+ .warning(
+ new FinalRClassEntriesWithOptimizedShrinkingDiagnostic(
+ holder.origin, field.getReference()));
+ }
+ }
+ }
+
private void analyzeClassInitializer(
RClassFieldToValueStore.Builder rClassValueBuilder, ProgramMethod programClassInitializer) {
IRCode code = programClassInitializer.buildIR(appView, MethodConversionOptions.nonConverting());
diff --git a/src/test/java/com/android/tools/r8/androidresources/DiagnosticOnFinalIntFieldsTest.java b/src/test/java/com/android/tools/r8/androidresources/DiagnosticOnFinalIntFieldsTest.java
new file mode 100644
index 0000000..a096a8f
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/androidresources/DiagnosticOnFinalIntFieldsTest.java
@@ -0,0 +1,83 @@
+// 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.androidresources;
+
+import static com.android.tools.r8.DiagnosticsMatcher.diagnosticType;
+
+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.androidresources.AndroidResourceTestingUtils.AndroidTestResource;
+import com.android.tools.r8.androidresources.AndroidResourceTestingUtils.AndroidTestResourceBuilder;
+import com.android.tools.r8.errors.FinalRClassEntriesWithOptimizedShrinkingDiagnostic;
+import com.google.common.collect.ImmutableList;
+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 DiagnosticOnFinalIntFieldsTest extends TestBase {
+
+ @Parameter(0)
+ public TestParameters parameters;
+
+ @Parameters(name = "{0}")
+ public static TestParametersCollection parameters() {
+ return getTestParameters().withDefaultDexRuntime().withAllApiLevels().build();
+ }
+
+ public static AndroidTestResource getTestResources(TemporaryFolder temp) throws Exception {
+ return new AndroidTestResourceBuilder()
+ .withSimpleManifestAndAppNameString()
+ .addRClassInitializeWithDefaultValues(R.string.class)
+ .build(temp);
+ }
+
+ @Test
+ public void testR8() throws Exception {
+ // We test the final id field type by simply passing the standard test resources but ignoring
+ // the aapt generated R class, instead we pass directly the R class from this file, which
+ // have no real resource references, but does have a non integer field.
+ testForR8(parameters.getBackend())
+ .setMinApi(parameters)
+ .addProgramClasses(FooBar.class)
+ .addAndroidResources(
+ getTestResources(temp),
+ temp.newFile("resout.zip").toPath(),
+ ImmutableList.of(ToolHelper.getClassAsBytes(R.string.class)))
+ .addKeepMainRule(FooBar.class)
+ .enableOptimizedShrinking()
+ .allowDiagnosticWarningMessages()
+ .compileWithExpectedDiagnostics(
+ diagnostics ->
+ diagnostics
+ .assertOnlyWarnings()
+ .assertWarningsMatch(
+ diagnosticType(FinalRClassEntriesWithOptimizedShrinkingDiagnostic.class)));
+ }
+
+ public static class FooBar {
+ public static void main(String[] args) {
+ if (System.currentTimeMillis() == 0) {
+ System.out.println(R.string.foo);
+ System.out.println(R.string.bar);
+ }
+ if (R.string.nonResource != null) {
+ System.out.println("bar");
+ }
+ }
+ }
+
+ public static class R {
+ public static class string {
+ private static Object nonResource = new Object();
+ public static int foo = 0x7f110004;
+ public static final int bar = 0x7f110005;
+ }
+ }
+}