[KeepAnno] Initial structure for a proto based keep specification
Bug: b/343389186
Change-Id: I639d5a64a25ffe2e92cf9e0073bb5084825baea4
diff --git a/d8_r8/commonBuildSrc/src/main/kotlin/DependenciesPlugin.kt b/d8_r8/commonBuildSrc/src/main/kotlin/DependenciesPlugin.kt
index 1f748ae..b54edc3 100644
--- a/d8_r8/commonBuildSrc/src/main/kotlin/DependenciesPlugin.kt
+++ b/d8_r8/commonBuildSrc/src/main/kotlin/DependenciesPlugin.kt
@@ -403,6 +403,7 @@
const val kotlinMetadataVersion = "2.0.0"
const val mockito = "2.10.0"
const val smaliVersion = "3.0.3"
+ const val protobufVersion = "3.19.3"
}
object Deps {
@@ -422,6 +423,7 @@
val mockito by lazy { "org.mockito:mockito-core:${Versions.mockito}" }
val smali by lazy { "com.android.tools.smali:smali:${Versions.smaliVersion}" }
val smaliUtil by lazy { "com.android.tools.smali:smali-util:${Versions.smaliVersion}" }
+ val protobuf by lazy { "com.google.protobuf:protobuf-java:${Versions.protobufVersion}" }
}
object ThirdPartyDeps {
diff --git a/d8_r8/keepanno/build.gradle.kts b/d8_r8/keepanno/build.gradle.kts
index a747444..f7c128b 100644
--- a/d8_r8/keepanno/build.gradle.kts
+++ b/d8_r8/keepanno/build.gradle.kts
@@ -51,7 +51,7 @@
dependencies {
compileOnly(Deps.asm)
compileOnly(Deps.guava)
- implementation("com.google.protobuf:protobuf-java:3.19.3")
+ compileOnly(Deps.protobuf)
}
tasks {
diff --git a/d8_r8/main/build.gradle.kts b/d8_r8/main/build.gradle.kts
index fd3282d..ed1c2c7 100644
--- a/d8_r8/main/build.gradle.kts
+++ b/d8_r8/main/build.gradle.kts
@@ -47,6 +47,7 @@
compileOnly(Deps.gson)
compileOnly(Deps.guava)
compileOnly(Deps.kotlinMetadata)
+ compileOnly(Deps.protobuf)
errorprone(Deps.errorprone)
}
diff --git a/d8_r8/resourceshrinker/build.gradle.kts b/d8_r8/resourceshrinker/build.gradle.kts
index db14c2f..3016524 100644
--- a/d8_r8/resourceshrinker/build.gradle.kts
+++ b/d8_r8/resourceshrinker/build.gradle.kts
@@ -36,9 +36,9 @@
dependencies {
compileOnly(Deps.asm)
compileOnly(Deps.guava)
+ compileOnly(Deps.protobuf)
compileOnly(files(resolve(ThirdPartyDeps.r8, "r8lib_8.2.20-dev.jar")))
implementation("com.android.tools.build:aapt2-proto:8.2.0-alpha10-10154469")
- implementation("com.google.protobuf:protobuf-java:3.19.3")
implementation("com.android.tools.layoutlib:layoutlib-api:31.5.0-alpha04")
implementation("com.android.tools:common:31.5.0-alpha04")
implementation("com.android.tools:sdk-common:31.5.0-alpha04")
diff --git a/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepDeclaration.java b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepDeclaration.java
index e80bfbd..3b32980 100644
--- a/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepDeclaration.java
+++ b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepDeclaration.java
@@ -3,10 +3,15 @@
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.keepanno.ast;
+import static com.android.tools.r8.keepanno.ast.KeepSpecUtils.doBuild;
+
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.Edge;
+import com.android.tools.r8.keepanno.utils.Unimplemented;
import java.util.function.Consumer;
import java.util.function.Function;
-/** Base class for the declarations represented in the keep annoations library. */
+/** Base class for the declarations represented in the keep annotations library. */
public abstract class KeepDeclaration {
public abstract KeepEdgeMetaInfo getMetaInfo();
@@ -47,4 +52,21 @@
public final int hashCode() {
throw new RuntimeException();
}
+
+ public final void buildDeclarationProto(KeepSpecProtos.Declaration.Builder builder) {
+ match(
+ edge -> doBuild(KeepSpecProtos.Edge.newBuilder(), edge::buildEdgeProto, builder::setEdge),
+ check -> {
+ throw new Unimplemented();
+ });
+ }
+
+ public static KeepDeclaration fromProto(
+ KeepSpecProtos.Declaration declaration, KeepSpecVersion version) {
+ if (declaration.hasEdge()) {
+ Edge edge = declaration.getEdge();
+ return KeepEdge.builder().applyProto(edge, version).build();
+ }
+ throw new Unimplemented();
+ }
}
diff --git a/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepEdge.java b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepEdge.java
index 6a741ce..a56f872 100644
--- a/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepEdge.java
+++ b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepEdge.java
@@ -3,7 +3,10 @@
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.keepanno.ast;
-import com.android.tools.r8.keepanno.proto.KeepAnnoProtos;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.Edge;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.MetaInfo;
+import com.android.tools.r8.keepanno.utils.Unimplemented;
/**
* An edge in the keep graph.
@@ -146,8 +149,9 @@
private Builder() {}
- public Builder applyProto(KeepAnnoProtos.Edge edge) {
+ public Builder applyProto(Edge edge, KeepSpecVersion version) {
// TODO(b/343389186): implement this.
+ KeepEdgeMetaInfo.builder().applyProto(edge.getMetaInfo(), version).build();
return this;
}
@@ -264,7 +268,8 @@
+ '}';
}
- public void buildProto(KeepAnnoProtos.Edge.Builder builder) {
- // TODO(b/343389186): implement this.
+ public void buildEdgeProto(KeepSpecProtos.Edge.Builder builder) {
+ KeepSpecUtils.doBuild(MetaInfo.newBuilder(), getMetaInfo()::buildProto, builder::setMetaInfo);
+ throw new Unimplemented();
}
}
diff --git a/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepEdgeMetaInfo.java b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepEdgeMetaInfo.java
index 9cf3305..2b2aa8d 100644
--- a/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepEdgeMetaInfo.java
+++ b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepEdgeMetaInfo.java
@@ -3,24 +3,24 @@
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.keepanno.ast;
+import static com.android.tools.r8.keepanno.ast.KeepSpecUtils.desc;
+
import com.android.tools.r8.keepanno.keeprules.RulePrintingUtils;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.Context;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.FieldDesc;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.MetaInfo;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.MethodDesc;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.TypeDesc;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
+import org.objectweb.asm.Type;
public class KeepEdgeMetaInfo {
- public enum KeepEdgeVersion {
- UNKNOWN;
-
- public String toVersionString() {
- return name();
- }
- }
-
private static final KeepEdgeMetaInfo NONE =
new KeepEdgeMetaInfo(
- KeepEdgeVersion.UNKNOWN, KeepEdgeContext.none(), KeepEdgeDescription.empty());
+ KeepSpecVersion.UNKNOWN, KeepEdgeContext.none(), KeepEdgeDescription.empty());
public static KeepEdgeMetaInfo none() {
return NONE;
@@ -30,12 +30,12 @@
return new Builder();
}
- private final KeepEdgeVersion version;
+ private final KeepSpecVersion version;
private final KeepEdgeContext context;
private final KeepEdgeDescription description;
private KeepEdgeMetaInfo(
- KeepEdgeVersion version, KeepEdgeContext context, KeepEdgeDescription description) {
+ KeepSpecVersion version, KeepEdgeContext context, KeepEdgeDescription description) {
this.version = version;
this.context = context;
this.description = description;
@@ -58,10 +58,10 @@
}
public boolean hasVersion() {
- return version != KeepEdgeVersion.UNKNOWN;
+ return version != KeepSpecVersion.UNKNOWN;
}
- public KeepEdgeVersion getVersion() {
+ public KeepSpecVersion getVersion() {
return version;
}
@@ -80,30 +80,61 @@
return "MetaInfo{" + String.join(", ", props) + "}";
}
+ public void buildProto(MetaInfo.Builder builder) {
+ KeepSpecUtils.doBuild(Context.newBuilder(), context::buildProto, builder::setContext);
+ if (!description.isEmpty()) {
+ builder.setDescription(description.description);
+ }
+ }
+
public static class Builder {
+ private KeepSpecVersion version = KeepSpecVersion.UNKNOWN;
private KeepEdgeContext context = KeepEdgeContext.none();
private KeepEdgeDescription description = KeepEdgeDescription.empty();
+ public Builder applyProto(MetaInfo metaInfo, KeepSpecVersion version) {
+ setVersion(version);
+ context = KeepEdgeContext.fromProto(metaInfo.getContext());
+ setDescription(metaInfo.getDescription());
+ return this;
+ }
+
+ public Builder setVersion(KeepSpecVersion version) {
+ this.version = version;
+ return this;
+ }
+
public Builder setDescription(String description) {
- this.description = new KeepEdgeDescription(description);
+ this.description =
+ description.isEmpty() ? KeepEdgeDescription.EMPTY : new KeepEdgeDescription(description);
+ return this;
+ }
+
+ public Builder setContext(KeepEdgeContext context) {
+ this.context = context;
return this;
}
public Builder setContextFromClassDescriptor(String classDescriptor) {
- context = new KeepEdgeClassContext(classDescriptor);
- return this;
+ return setContext(new KeepEdgeClassContext(classDescriptor));
}
public Builder setContextFromMethodDescriptor(
String classDescriptor, String methodName, String methodDescriptor) {
- context = new KeepEdgeMethodContext(classDescriptor, methodName, methodDescriptor);
- return this;
+ Type methodType = Type.getMethodType(methodDescriptor);
+ Type[] argumentTypes = methodType.getArgumentTypes();
+ List<String> parameters = new ArrayList<>(argumentTypes.length);
+ for (Type argumentType : argumentTypes) {
+ parameters.add(argumentType.getDescriptor());
+ }
+ return setContext(
+ new KeepEdgeMethodContext(
+ classDescriptor, methodName, methodType.getReturnType().getDescriptor(), parameters));
}
public Builder setContextFromFieldDescriptor(
String classDescriptor, String fieldName, String fieldType) {
- context = new KeepEdgeFieldContext(classDescriptor, fieldName, fieldType);
- return this;
+ return setContext(new KeepEdgeFieldContext(classDescriptor, fieldName, fieldType));
}
public KeepEdgeMetaInfo build() {
@@ -111,7 +142,7 @@
&& description.equals(KeepEdgeDescription.empty())) {
return none();
}
- return new KeepEdgeMetaInfo(KeepEdgeVersion.UNKNOWN, context, description);
+ return new KeepEdgeMetaInfo(version, context, description);
}
}
@@ -124,6 +155,32 @@
private KeepEdgeContext() {}
+ public static KeepEdgeContext fromProto(Context context) {
+ if (context.hasClassDesc()) {
+ return new KeepEdgeClassContext(context.getClassDesc().getDesc());
+ }
+ if (context.hasMethodDesc()) {
+ MethodDesc methodDesc = context.getMethodDesc();
+ List<String> parameters = new ArrayList<>(methodDesc.getParameterTypesCount());
+ for (TypeDesc typeDesc : methodDesc.getParameterTypesList()) {
+ parameters.add(typeDesc.getDesc());
+ }
+ return new KeepEdgeMethodContext(
+ methodDesc.getHolder().getDesc(),
+ methodDesc.getName(),
+ methodDesc.getReturnType().getDesc(),
+ parameters);
+ }
+ if (context.hasFieldDesc()) {
+ FieldDesc fieldDesc = context.getFieldDesc();
+ return new KeepEdgeFieldContext(
+ fieldDesc.getHolder().getDesc(),
+ fieldDesc.getName(),
+ fieldDesc.getFieldType().getDesc());
+ }
+ return none();
+ }
+
public String getDescriptorString() {
throw new KeepEdgeException("Invalid attempt to get descriptor string from none context");
}
@@ -137,6 +194,10 @@
public int hashCode() {
return System.identityHashCode(this);
}
+
+ public void buildProto(Context.Builder builder) {
+ assert this == none();
+ }
}
private static class KeepEdgeClassContext extends KeepEdgeContext {
@@ -168,26 +229,41 @@
public int hashCode() {
return classDescriptor.hashCode();
}
+
+ @Override
+ public void buildProto(Context.Builder builder) {
+ builder.setClassDesc(desc(classDescriptor));
+ }
}
private static class KeepEdgeMethodContext extends KeepEdgeContext {
private final String classDescriptor;
private final String methodName;
- private final String methodDescriptor;
+ private final String methodReturnType;
+ private final List<String> methodParameters;
public KeepEdgeMethodContext(
- String classDescriptor, String methodName, String methodDescriptor) {
+ String classDescriptor,
+ String methodName,
+ String methodReturnType,
+ List<String> methodParameters) {
assert classDescriptor != null;
assert methodName != null;
- assert methodDescriptor != null;
+ assert methodParameters != null;
this.classDescriptor = classDescriptor;
this.methodName = methodName;
- this.methodDescriptor = methodDescriptor;
+ this.methodReturnType = methodReturnType;
+ this.methodParameters = methodParameters;
}
@Override
public String getDescriptorString() {
- return classDescriptor + methodName + methodDescriptor;
+ StringBuilder builder = new StringBuilder();
+ builder.append(classDescriptor).append(methodName).append('(');
+ for (String parameter : methodParameters) {
+ builder.append(parameter);
+ }
+ return builder.append(')').append(methodReturnType).toString();
}
@Override
@@ -201,12 +277,26 @@
KeepEdgeMethodContext that = (KeepEdgeMethodContext) o;
return classDescriptor.equals(that.classDescriptor)
&& methodName.equals(that.methodName)
- && methodDescriptor.equals(that.methodDescriptor);
+ && methodReturnType.equals(that.methodReturnType)
+ && methodParameters.equals(that.methodParameters);
}
@Override
public int hashCode() {
- return Objects.hash(classDescriptor, methodName, methodDescriptor);
+ return Objects.hash(classDescriptor, methodName, methodReturnType, methodParameters);
+ }
+
+ @Override
+ public void buildProto(Context.Builder builder) {
+ MethodDesc.Builder methodBuilder =
+ MethodDesc.newBuilder()
+ .setHolder(desc(classDescriptor))
+ .setName(methodName)
+ .setReturnType(desc(methodReturnType));
+ for (String methodParameter : methodParameters) {
+ methodBuilder.addParameterTypes(desc(methodParameter));
+ }
+ builder.setMethodDesc(methodBuilder.build());
}
}
@@ -244,6 +334,16 @@
public int hashCode() {
return Objects.hash(classDescriptor, fieldName, fieldType);
}
+
+ @Override
+ public void buildProto(Context.Builder builder) {
+ builder.setFieldDesc(
+ FieldDesc.newBuilder()
+ .setHolder(desc(classDescriptor))
+ .setName(fieldName)
+ .setFieldType(desc(fieldType))
+ .build());
+ }
}
private static class KeepEdgeDescription {
@@ -261,12 +361,11 @@
}
@Override
- @SuppressWarnings("EqualsGetClass")
public boolean equals(Object o) {
if (this == o) {
return true;
}
- if (o == null || getClass() != o.getClass()) {
+ if (!(o instanceof KeepEdgeDescription)) {
return false;
}
KeepEdgeDescription that = (KeepEdgeDescription) o;
@@ -282,5 +381,9 @@
public String toString() {
return description;
}
+
+ public boolean isEmpty() {
+ return description.isEmpty();
+ }
}
}
diff --git a/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepSpecUtils.java b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepSpecUtils.java
new file mode 100644
index 0000000..2e45c4c
--- /dev/null
+++ b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepSpecUtils.java
@@ -0,0 +1,29 @@
+// 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.keepanno.ast;
+
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.TypeDesc;
+import com.google.protobuf.MessageOrBuilder;
+import java.util.function.Consumer;
+
+public final class KeepSpecUtils {
+
+ private KeepSpecUtils() {}
+
+ public static <T extends MessageOrBuilder> void doBuild(
+ T builder, Consumer<T> buildMethod, Consumer<T> setter) {
+ buildMethod.accept(builder);
+ setter.accept(builder);
+ }
+
+ public static <T extends MessageOrBuilder> String toString(T builder, Consumer<T> buildMethod) {
+ buildMethod.accept(builder);
+ return builder.toString();
+ }
+
+ public static TypeDesc desc(String descriptor) {
+ return TypeDesc.newBuilder().setDesc(descriptor).build();
+ }
+}
diff --git a/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepSpecVersion.java b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepSpecVersion.java
new file mode 100644
index 0000000..c9034a6
--- /dev/null
+++ b/src/keepanno/java/com/android/tools/r8/keepanno/ast/KeepSpecVersion.java
@@ -0,0 +1,48 @@
+// 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.keepanno.ast;
+
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.Version;
+
+public enum KeepSpecVersion {
+ UNKNOWN(0, 0, 0),
+ ALPHA(0, 1, 0);
+
+ private final int major;
+ private final int minor;
+ private final int patch;
+
+ KeepSpecVersion(int major, int minor, int patch) {
+ this.major = major;
+ this.minor = minor;
+ this.patch = patch;
+ }
+
+ public static KeepSpecVersion getCurrent() {
+ return ALPHA;
+ }
+
+ public static KeepSpecVersion fromProto(Version version) {
+ for (KeepSpecVersion value : values()) {
+ if (value.major == version.getMajor()
+ && value.minor == version.getMinor()
+ && value.patch == version.getPatch()) {
+ return value;
+ }
+ }
+ return UNKNOWN;
+ }
+
+ public String toVersionString() {
+ return "" + major + "." + minor + "." + patch;
+ }
+
+ public void buildProto(KeepSpecProtos.Version.Builder builder) {
+ builder.setMajor(major);
+ builder.setMinor(minor);
+ builder.setPatch(patch);
+ }
+}
diff --git a/src/keepanno/proto/keepanno.proto b/src/keepanno/proto/keepanno.proto
deleted file mode 100644
index a4f96d7..0000000
--- a/src/keepanno/proto/keepanno.proto
+++ /dev/null
@@ -1,56 +0,0 @@
-// 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.
-syntax = "proto3";
-
-package com.android.tools.r8.keepanno.proto;
-
-// All messages are placed under the outer class. This makes it a bit nicer to
-// implement the AST <-> Proto conversions without type conflicts.
-option java_multiple_files = false;
-
-// Descriptive name of the outer class (default would have been `Keepanno`).
-option java_outer_classname = "KeepAnnoProtos";
-
-// Java package consistent with R8 convention.
-option java_package = "com.android.tools.r8.keepanno.proto";
-
-message Version {
- uint32 major = 1;
- uint32 minor = 2;
- uint32 patch = 3;
-}
-
-message Context {
- oneof context_oneof {
- string class_descriptor = 1;
- string method_descriptor = 2;
- string field_descriptor = 3;
- }
-}
-
-message MetaInfo {
- Version version = 1;
- Context context = 2;
-}
-
-message Declaration {
- MetaInfo meta_info = 1;
- oneof decl_oneof {
- Edge edge = 2;
- CheckRemoved check_removed = 3;
- CheckDiscarded check_discarded = 4;
- }
-}
-
-message CheckRemoved {
- // TODO(b/343389186): Add content.
-}
-
-message CheckDiscarded {
- // TODO(b/343389186): Add content.
-}
-
-message Edge {
- // TODO(b/343389186): Add content.
-}
diff --git a/src/keepanno/proto/keepspec.proto b/src/keepanno/proto/keepspec.proto
new file mode 100644
index 0000000..b5d5fd3
--- /dev/null
+++ b/src/keepanno/proto/keepspec.proto
@@ -0,0 +1,85 @@
+// 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.
+syntax = "proto3";
+
+package com.android.tools.r8.keepanno.proto;
+
+// All messages are placed under the outer class. This makes it a bit nicer to
+// implement the AST <-> Proto conversions without type conflicts.
+option java_multiple_files = false;
+
+// Camel-case the outer class name (default is `Keepspec`).
+option java_outer_classname = "KeepSpecProtos";
+
+// Java package consistent with R8 convention.
+option java_package = "com.android.tools.r8.keepanno.proto";
+
+// Top-level container for the keep specification
+message KeepSpec {
+ Version version = 1;
+ repeated Declaration declarations = 2;
+}
+
+message Version {
+ uint32 major = 1;
+ uint32 minor = 2;
+ uint32 patch = 3;
+}
+
+message Declaration {
+ oneof decl_oneof {
+ Edge edge = 2;
+ CheckRemoved check_removed = 3;
+ CheckOptimizedOut check_optimized_out = 4;
+ }
+}
+
+// Note: the messages and fields avoid the use of `descriptor` in any place as
+// that name is used internally in the protobuf encodings. We consistently use
+// the short-form `desc` throughout.
+
+message Context {
+ oneof context_oneof {
+ TypeDesc class_desc = 1;
+ MethodDesc method_desc = 2;
+ FieldDesc field_desc = 3;
+ }
+}
+
+message TypeDesc {
+ string desc = 1;
+}
+
+message MethodDesc {
+ string name = 1;
+ TypeDesc holder = 2;
+ TypeDesc returnType = 3;
+ repeated TypeDesc parameterTypes = 4;
+}
+
+message FieldDesc {
+ string name = 1;
+ TypeDesc holder = 2;
+ TypeDesc fieldType = 3;
+}
+
+message MetaInfo {
+ Context context = 1;
+ string description = 2;
+}
+
+message CheckRemoved {
+ MetaInfo meta_info = 1;
+ // TODO(b/343389186): Add content.
+}
+
+message CheckOptimizedOut {
+ MetaInfo meta_info = 1;
+ // TODO(b/343389186): Add content.
+}
+
+message Edge {
+ MetaInfo meta_info = 1;
+ // TODO(b/343389186): Add content.
+}
diff --git a/src/main/java/com/android/tools/r8/R8.java b/src/main/java/com/android/tools/r8/R8.java
index ce2e296..74f9c92 100644
--- a/src/main/java/com/android/tools/r8/R8.java
+++ b/src/main/java/com/android/tools/r8/R8.java
@@ -94,6 +94,7 @@
import com.android.tools.r8.shaking.Enqueuer.Mode;
import com.android.tools.r8.shaking.EnqueuerFactory;
import com.android.tools.r8.shaking.EnqueuerResult;
+import com.android.tools.r8.shaking.KeepSpecificationSource;
import com.android.tools.r8.shaking.MainDexInfo;
import com.android.tools.r8.shaking.MainDexListBuilder;
import com.android.tools.r8.shaking.ProguardConfigurationRule;
@@ -310,6 +311,8 @@
timing.begin("Register references and more setup");
assert ArtProfileCompletenessChecker.verify(appView);
+ readKeepSpecifications(appView, keepDeclarations);
+
// Check for potentially having pass-through of Cf-code for kotlin libraries.
options.enableCfByteCodePassThrough =
options.isGeneratingClassFiles() && KotlinMetadataUtils.mayProcessKotlinMetadata(appView);
@@ -894,6 +897,19 @@
}
}
+ private void readKeepSpecifications(
+ AppView<AppInfoWithClassHierarchy> appView, List<KeepDeclaration> keepDeclarations) {
+ timing.begin("Read keep specifications");
+ try {
+ for (KeepSpecificationSource source : appView.options().getKeepSpecifications()) {
+ source.parse(keepDeclarations::add);
+ }
+ } catch (ResourceException e) {
+ options.reporter.error(new ExceptionDiagnostic(e, e.getOrigin()));
+ }
+ timing.end();
+ }
+
private void writeKeepDeclarationsToConfigurationConsumer(
List<KeepDeclaration> keepDeclarations) {
if (options.configurationConsumer == null) {
diff --git a/src/main/java/com/android/tools/r8/R8Command.java b/src/main/java/com/android/tools/r8/R8Command.java
index adfaf29..36f1ff3 100644
--- a/src/main/java/com/android/tools/r8/R8Command.java
+++ b/src/main/java/com/android/tools/r8/R8Command.java
@@ -27,6 +27,7 @@
import com.android.tools.r8.origin.PathOrigin;
import com.android.tools.r8.profile.art.ArtProfileForRewriting;
import com.android.tools.r8.shaking.FilteredClassPath;
+import com.android.tools.r8.shaking.KeepSpecificationSource;
import com.android.tools.r8.shaking.ProguardConfiguration;
import com.android.tools.r8.shaking.ProguardConfigurationParser;
import com.android.tools.r8.shaking.ProguardConfigurationParserOptions;
@@ -120,6 +121,7 @@
private Consumer<ProguardConfiguration.Builder> proguardConfigurationConsumerForTesting = null;
private Consumer<List<ProguardConfigurationRule>> syntheticProguardRulesConsumer = null;
private StringConsumer desugaredLibraryKeepRuleConsumer = null;
+ private final List<KeepSpecificationSource> keepSpecifications = new ArrayList<>();
private final List<ProguardConfigurationSource> proguardConfigs = new ArrayList<>();
private boolean disableTreeShaking = false;
private boolean disableMinification = false;
@@ -235,6 +237,20 @@
return self();
}
+ public Builder addKeepSpecificationFiles(Path... paths) {
+ return addKeepSpecificationFiles(Arrays.asList(paths));
+ }
+
+ public Builder addKeepSpecificationFiles(Collection<Path> paths) {
+ paths.forEach(p -> keepSpecifications.add(KeepSpecificationSource.fromFile(p)));
+ return self();
+ }
+
+ public Builder addKeepSpecificationData(byte[] data, Origin origin) {
+ keepSpecifications.add(KeepSpecificationSource.fromBytes(origin, data));
+ return self();
+ }
+
/**
* Set Proguard compatibility mode.
*
@@ -755,7 +771,8 @@
getCancelCompilationChecker(),
androidResourceProvider,
androidResourceConsumer,
- resourceShrinkerConfiguration);
+ resourceShrinkerConfiguration,
+ keepSpecifications);
if (inputDependencyGraphConsumer != null) {
inputDependencyGraphConsumer.finished();
@@ -933,6 +950,7 @@
private final List<ProguardConfigurationRule> mainDexKeepRules;
private final ProguardConfiguration proguardConfiguration;
+ private final List<KeepSpecificationSource> keepSpecifications;
private final boolean enableTreeShaking;
private final boolean enableMinification;
private final boolean forceProguardCompatibility;
@@ -1051,7 +1069,8 @@
CancelCompilationChecker cancelCompilationChecker,
AndroidResourceProvider androidResourceProvider,
AndroidResourceConsumer androidResourceConsumer,
- ResourceShrinkerConfiguration resourceShrinkerConfiguration) {
+ ResourceShrinkerConfiguration resourceShrinkerConfiguration,
+ List<KeepSpecificationSource> keepSpecifications) {
super(
inputApp,
mode,
@@ -1078,6 +1097,7 @@
assert mainDexKeepRules != null;
this.mainDexKeepRules = mainDexKeepRules;
this.proguardConfiguration = proguardConfiguration;
+ this.keepSpecifications = keepSpecifications;
this.enableTreeShaking = enableTreeShaking;
this.enableMinification = enableMinification;
this.forceProguardCompatibility = forceProguardCompatibility;
@@ -1105,6 +1125,7 @@
super(printHelp, printVersion);
mainDexKeepRules = ImmutableList.of();
proguardConfiguration = null;
+ keepSpecifications = null;
enableTreeShaking = false;
enableMinification = false;
forceProguardCompatibility = false;
@@ -1164,6 +1185,8 @@
&& !internal.isShrinking()
&& !internal.isMinifying());
+ internal.setKeepSpecificationSources(keepSpecifications);
+
assert !internal.verbose;
internal.mainDexKeepRules = mainDexKeepRules;
internal.minimalMainDex = internal.debug;
diff --git a/src/main/java/com/android/tools/r8/shaking/KeepSpecificationSource.java b/src/main/java/com/android/tools/r8/shaking/KeepSpecificationSource.java
new file mode 100644
index 0000000..b7c4fe8
--- /dev/null
+++ b/src/main/java/com/android/tools/r8/shaking/KeepSpecificationSource.java
@@ -0,0 +1,91 @@
+// 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.shaking;
+
+import com.android.tools.r8.ResourceException;
+import com.android.tools.r8.keepanno.ast.KeepDeclaration;
+import com.android.tools.r8.keepanno.ast.KeepSpecVersion;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.Declaration;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.KeepSpec;
+import com.android.tools.r8.origin.Origin;
+import com.android.tools.r8.origin.PathOrigin;
+import com.google.protobuf.InvalidProtocolBufferException;
+import java.io.IOException;
+import java.io.InputStream;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.util.function.Consumer;
+
+public abstract class KeepSpecificationSource {
+
+ public static KeepSpecificationSource fromFile(Path path) {
+ return new KeepSpecificationFile(path);
+ }
+
+ public static KeepSpecificationSource fromBytes(Origin origin, byte[] bytes) {
+ return new KeepSpecificationBytes(origin, bytes);
+ }
+
+ private final Origin origin;
+
+ private KeepSpecificationSource(Origin origin) {
+ this.origin = origin;
+ }
+
+ public Origin getOrigin() {
+ return origin;
+ }
+
+ public void parse(Consumer<KeepDeclaration> consumer) throws ResourceException {
+ KeepSpec spec = read();
+ KeepSpecVersion version = KeepSpecVersion.fromProto(spec.getVersion());
+ if (version == KeepSpecVersion.UNKNOWN) {
+ throw new ResourceException(getOrigin(), "Unknown keepspec version " + spec.getVersion());
+ }
+ for (Declaration declaration : spec.getDeclarationsList()) {
+ consumer.accept(KeepDeclaration.fromProto(declaration, version));
+ }
+ }
+
+ abstract KeepSpec read() throws ResourceException;
+
+ private static class KeepSpecificationFile extends KeepSpecificationSource {
+
+ private final Path path;
+
+ private KeepSpecificationFile(Path path) {
+ super(new PathOrigin(path));
+ this.path = path;
+ }
+
+ @Override
+ KeepSpec read() throws ResourceException {
+ try (InputStream stream = Files.newInputStream(path)) {
+ return KeepSpec.parseFrom(stream);
+ } catch (IOException e) {
+ throw new ResourceException(getOrigin(), e);
+ }
+ }
+ }
+
+ private static class KeepSpecificationBytes extends KeepSpecificationSource {
+
+ private final byte[] content;
+
+ public KeepSpecificationBytes(Origin origin, byte[] bytes) {
+ super(origin);
+ this.content = bytes;
+ }
+
+ @Override
+ KeepSpec read() throws ResourceException {
+ try {
+ return KeepSpec.parseFrom(content);
+ } catch (InvalidProtocolBufferException e) {
+ throw new ResourceException(getOrigin(), e);
+ }
+ }
+ }
+}
diff --git a/src/main/java/com/android/tools/r8/utils/InternalOptions.java b/src/main/java/com/android/tools/r8/utils/InternalOptions.java
index b3d395b..ef49e3a 100644
--- a/src/main/java/com/android/tools/r8/utils/InternalOptions.java
+++ b/src/main/java/com/android/tools/r8/utils/InternalOptions.java
@@ -100,6 +100,7 @@
import com.android.tools.r8.shaking.AppInfoWithLiveness;
import com.android.tools.r8.shaking.Enqueuer;
import com.android.tools.r8.shaking.GlobalKeepInfoConfiguration;
+import com.android.tools.r8.shaking.KeepSpecificationSource;
import com.android.tools.r8.shaking.ProguardConfiguration;
import com.android.tools.r8.shaking.ProguardConfigurationRule;
import com.android.tools.r8.threading.ThreadingModule;
@@ -118,6 +119,7 @@
import java.io.IOException;
import java.nio.file.Paths;
import java.util.ArrayList;
+import java.util.Collection;
import java.util.Collections;
import java.util.Comparator;
import java.util.Deque;
@@ -240,6 +242,8 @@
private final ProguardConfiguration proguardConfiguration;
public final Reporter reporter;
+ private Collection<KeepSpecificationSource> keepSpecificationSources = Collections.emptyList();
+
// TODO(zerny): Make this private-final once we have full program-consumer support.
public ProgramConsumer programConsumer = null;
@@ -314,6 +318,14 @@
return lazyThreadingModule;
}
+ public Collection<KeepSpecificationSource> getKeepSpecifications() {
+ return keepSpecificationSources;
+ }
+
+ public void setKeepSpecificationSources(Collection<KeepSpecificationSource> specifications) {
+ keepSpecificationSources = specifications;
+ }
+
private void keepDebugRelatedInformation() {
assert !proguardConfiguration.isObfuscating();
getProguardConfiguration().getKeepAttributes().sourceFile = true;
diff --git a/src/test/java/com/android/tools/r8/keepanno/KeepAnnoTestBuilder.java b/src/test/java/com/android/tools/r8/keepanno/KeepAnnoTestBuilder.java
index 3a4a3c4..5d93d82 100644
--- a/src/test/java/com/android/tools/r8/keepanno/KeepAnnoTestBuilder.java
+++ b/src/test/java/com/android/tools/r8/keepanno/KeepAnnoTestBuilder.java
@@ -23,7 +23,13 @@
import com.android.tools.r8.keepanno.asm.KeepEdgeReader;
import com.android.tools.r8.keepanno.asm.KeepEdgeWriter;
import com.android.tools.r8.keepanno.ast.KeepDeclaration;
+import com.android.tools.r8.keepanno.ast.KeepSpecUtils;
+import com.android.tools.r8.keepanno.ast.KeepSpecVersion;
import com.android.tools.r8.keepanno.keeprules.KeepRuleExtractorOptions;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.Declaration;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.KeepSpec;
+import com.android.tools.r8.keepanno.proto.KeepSpecProtos.Version;
+import com.android.tools.r8.keepanno.utils.Unimplemented;
import com.android.tools.r8.origin.Origin;
import com.android.tools.r8.utils.DescriptorUtils;
import com.android.tools.r8.utils.InternalOptions;
@@ -280,6 +286,29 @@
if (isNormalizeEdges()) {
List<KeepDeclaration> declarations = KeepEdgeReader.readKeepEdges(classFileData);
if (!declarations.isEmpty()) {
+ List<KeepDeclaration> legacyExtract = new ArrayList<>();
+ KeepSpec.Builder keepSpecBuilder = KeepSpec.newBuilder();
+ KeepSpecUtils.doBuild(
+ Version.newBuilder(),
+ KeepSpecVersion.getCurrent()::buildProto,
+ keepSpecBuilder::setVersion);
+ for (KeepDeclaration declaration : declarations) {
+ try {
+ KeepSpecUtils.doBuild(
+ Declaration.newBuilder(),
+ declaration::buildDeclarationProto,
+ keepSpecBuilder::addDeclarations);
+ } catch (Unimplemented e) {
+ legacyExtract.add(declaration);
+ }
+ }
+ builder
+ .getBuilder()
+ .addKeepSpecificationData(keepSpecBuilder.build().toByteArray(), Origin.unknown());
+ if (legacyExtract.isEmpty()) {
+ // TODO(b/343389186): Finish the proto encoding and remove the below extraction.
+ return;
+ }
String binaryName =
DescriptorUtils.getBinaryNameFromDescriptor(extractClassDescriptor(classFileData));
String synthesizingTarget = binaryName + "$$ExtractedKeepEdges";
@@ -292,7 +321,7 @@
"java/lang/Object",
null);
KeepEdgeWriter.writeExtractedEdges(
- declarations,
+ legacyExtract,
(descriptor, visible) ->
KeepAnnoTestUtils.wrap(classWriter.visitAnnotation(descriptor, visible)));
classWriter.visitEnd();