blob: bb8dd5f8549d37f92ccf110240859aa2e3bb226b [file] [log] [blame]
// Copyright (c) 2022, 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.asm;
import com.android.tools.r8.keepanno.annotations.KeepConstants;
import com.android.tools.r8.keepanno.annotations.KeepConstants.Binding;
import com.android.tools.r8.keepanno.annotations.KeepConstants.Condition;
import com.android.tools.r8.keepanno.annotations.KeepConstants.Edge;
import com.android.tools.r8.keepanno.annotations.KeepConstants.Item;
import com.android.tools.r8.keepanno.annotations.KeepConstants.Option;
import com.android.tools.r8.keepanno.annotations.KeepConstants.Target;
import com.android.tools.r8.keepanno.ast.KeepBindings;
import com.android.tools.r8.keepanno.ast.KeepClassReference;
import com.android.tools.r8.keepanno.ast.KeepCondition;
import com.android.tools.r8.keepanno.ast.KeepConsequences;
import com.android.tools.r8.keepanno.ast.KeepEdge;
import com.android.tools.r8.keepanno.ast.KeepEdgeException;
import com.android.tools.r8.keepanno.ast.KeepEdgeMetaInfo;
import com.android.tools.r8.keepanno.ast.KeepExtendsPattern;
import com.android.tools.r8.keepanno.ast.KeepFieldNamePattern;
import com.android.tools.r8.keepanno.ast.KeepFieldPattern;
import com.android.tools.r8.keepanno.ast.KeepFieldTypePattern;
import com.android.tools.r8.keepanno.ast.KeepItemPattern;
import com.android.tools.r8.keepanno.ast.KeepMemberPattern;
import com.android.tools.r8.keepanno.ast.KeepMethodNamePattern;
import com.android.tools.r8.keepanno.ast.KeepMethodParametersPattern;
import com.android.tools.r8.keepanno.ast.KeepMethodPattern;
import com.android.tools.r8.keepanno.ast.KeepMethodReturnTypePattern;
import com.android.tools.r8.keepanno.ast.KeepOptions;
import com.android.tools.r8.keepanno.ast.KeepOptions.KeepOption;
import com.android.tools.r8.keepanno.ast.KeepPreconditions;
import com.android.tools.r8.keepanno.ast.KeepQualifiedClassNamePattern;
import com.android.tools.r8.keepanno.ast.KeepTarget;
import com.android.tools.r8.keepanno.ast.KeepTypePattern;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.function.Consumer;
import org.objectweb.asm.AnnotationVisitor;
import org.objectweb.asm.ClassReader;
import org.objectweb.asm.ClassVisitor;
import org.objectweb.asm.FieldVisitor;
import org.objectweb.asm.MethodVisitor;
import org.objectweb.asm.Opcodes;
import org.objectweb.asm.Type;
public class KeepEdgeReader implements Opcodes {
public static int ASM_VERSION = ASM9;
public static Set<KeepEdge> readKeepEdges(byte[] classFileBytes) {
ClassReader reader = new ClassReader(classFileBytes);
Set<KeepEdge> edges = new HashSet<>();
reader.accept(new KeepEdgeClassVisitor(edges::add), ClassReader.SKIP_CODE);
return edges;
}
private static class KeepEdgeClassVisitor extends ClassVisitor {
private final Parent<KeepEdge> parent;
private String className;
KeepEdgeClassVisitor(Parent<KeepEdge> parent) {
super(ASM_VERSION);
this.parent = parent;
}
private static String binaryNameToTypeName(String binaryName) {
return binaryName.replace('/', '.');
}
@Override
public void visit(
int version,
int access,
String name,
String signature,
String superName,
String[] interfaces) {
super.visit(version, access, name, signature, superName, interfaces);
this.className = binaryNameToTypeName(name);
}
@Override
public AnnotationVisitor visitAnnotation(String descriptor, boolean visible) {
// Skip any visible annotations as @KeepEdge is not runtime visible.
if (visible) {
return null;
}
if (descriptor.equals(Edge.DESCRIPTOR)) {
return new KeepEdgeVisitor(parent, this::setContext);
}
if (descriptor.equals(KeepConstants.UsesReflection.DESCRIPTOR)) {
KeepItemPattern classItem =
KeepItemPattern.builder()
.setClassPattern(KeepQualifiedClassNamePattern.exact(className))
.build();
return new UsesReflectionVisitor(parent, this::setContext, classItem);
}
return null;
}
private void setContext(KeepEdgeMetaInfo.Builder builder) {
builder.setContextFromClassDescriptor(KeepEdgeReaderUtils.javaTypeToDescriptor(className));
}
@Override
public MethodVisitor visitMethod(
int access, String name, String descriptor, String signature, String[] exceptions) {
return new KeepEdgeMethodVisitor(parent, className, name, descriptor);
}
@Override
public FieldVisitor visitField(
int access, String name, String descriptor, String signature, Object value) {
return new KeepEdgeFieldVisitor(parent, className, name, descriptor);
}
}
private static class KeepEdgeMethodVisitor extends MethodVisitor {
private final Parent<KeepEdge> parent;
private final String className;
private final String methodName;
private final String methodDescriptor;
KeepEdgeMethodVisitor(
Parent<KeepEdge> parent, String className, String methodName, String methodDescriptor) {
super(ASM_VERSION);
this.parent = parent;
this.className = className;
this.methodName = methodName;
this.methodDescriptor = methodDescriptor;
}
private KeepItemPattern createItemContext() {
String returnTypeDescriptor = Type.getReturnType(methodDescriptor).getDescriptor();
Type[] argumentTypes = Type.getArgumentTypes(methodDescriptor);
KeepMethodParametersPattern.Builder builder = KeepMethodParametersPattern.builder();
for (Type type : argumentTypes) {
builder.addParameterTypePattern(KeepTypePattern.fromDescriptor(type.getDescriptor()));
}
KeepMethodReturnTypePattern returnTypePattern =
"V".equals(returnTypeDescriptor)
? KeepMethodReturnTypePattern.voidType()
: KeepMethodReturnTypePattern.fromType(
KeepTypePattern.fromDescriptor(returnTypeDescriptor));
return KeepItemPattern.builder()
.setClassPattern(KeepQualifiedClassNamePattern.exact(className))
.setMemberPattern(
KeepMethodPattern.builder()
.setNamePattern(KeepMethodNamePattern.exact(methodName))
.setReturnTypePattern(returnTypePattern)
.setParametersPattern(builder.build())
.build())
.build();
}
@Override
public AnnotationVisitor visitAnnotation(String descriptor, boolean visible) {
// Skip any visible annotations as @KeepEdge is not runtime visible.
if (visible) {
return null;
}
if (descriptor.equals(Edge.DESCRIPTOR)) {
return new KeepEdgeVisitor(parent, this::setContext);
}
if (descriptor.equals(KeepConstants.UsesReflection.DESCRIPTOR)) {
return new UsesReflectionVisitor(parent, this::setContext, createItemContext());
}
return null;
}
private void setContext(KeepEdgeMetaInfo.Builder builder) {
builder.setContextFromMethodDescriptor(
KeepEdgeReaderUtils.javaTypeToDescriptor(className), methodName, methodDescriptor);
}
}
private static class KeepEdgeFieldVisitor extends FieldVisitor {
private final Parent<KeepEdge> parent;
private final String className;
private final String fieldName;
private final String fieldDescriptor;
KeepEdgeFieldVisitor(
Parent<KeepEdge> parent, String className, String fieldName, String fieldDescriptor) {
super(ASM_VERSION);
this.parent = parent;
this.className = className;
this.fieldName = fieldName;
this.fieldDescriptor = fieldDescriptor;
}
private KeepItemPattern createItemContext() {
KeepFieldTypePattern typePattern =
KeepFieldTypePattern.fromType(KeepTypePattern.fromDescriptor(fieldDescriptor));
return KeepItemPattern.builder()
.setClassPattern(KeepQualifiedClassNamePattern.exact(className))
.setMemberPattern(
KeepFieldPattern.builder()
.setNamePattern(KeepFieldNamePattern.exact(fieldName))
.setTypePattern(typePattern)
.build())
.build();
}
private void setContext(KeepEdgeMetaInfo.Builder builder) {
builder.setContextFromFieldDescriptor(
KeepEdgeReaderUtils.javaTypeToDescriptor(className), fieldName, fieldDescriptor);
}
@Override
public AnnotationVisitor visitAnnotation(String descriptor, boolean visible) {
// Skip any visible annotations as @KeepEdge is not runtime visible.
if (visible) {
return null;
}
if (descriptor.equals(Edge.DESCRIPTOR)) {
return new KeepEdgeVisitor(parent, this::setContext);
}
if (descriptor.equals(KeepConstants.UsesReflection.DESCRIPTOR)) {
return new UsesReflectionVisitor(parent, this::setContext, createItemContext());
}
return null;
}
}
// Interface for providing AST result(s) for a sub-tree back up to its parent.
private interface Parent<T> {
void accept(T result);
}
private abstract static class AnnotationVisitorBase extends AnnotationVisitor {
AnnotationVisitorBase() {
super(ASM_VERSION);
}
@Override
public void visit(String name, Object value) {
throw new KeepEdgeException("Unexpected value in @KeepEdge: " + name + " = " + value);
}
@Override
public AnnotationVisitor visitAnnotation(String name, String descriptor) {
throw new KeepEdgeException("Unexpected annotation in @KeepEdge: " + name);
}
@Override
public void visitEnum(String name, String descriptor, String value) {
throw new KeepEdgeException("Unexpected enum in @KeepEdge: " + name);
}
@Override
public AnnotationVisitor visitArray(String name) {
throw new KeepEdgeException("Unexpected array in @KeepEdge: " + name);
}
}
private static class KeepEdgeVisitor extends AnnotationVisitorBase {
private final Parent<KeepEdge> parent;
private final KeepEdge.Builder builder = KeepEdge.builder();
private final KeepEdgeMetaInfo.Builder metaInfoBuilder = KeepEdgeMetaInfo.builder();
KeepEdgeVisitor(Parent<KeepEdge> parent, Consumer<KeepEdgeMetaInfo.Builder> addContext) {
this.parent = parent;
addContext.accept(metaInfoBuilder);
}
@Override
public void visit(String name, Object value) {
if (name.equals(Edge.description) && value instanceof String) {
metaInfoBuilder.setDescription((String) value);
return;
}
super.visit(name, value);
}
@Override
public AnnotationVisitor visitArray(String name) {
if (name.equals(Edge.bindings)) {
return new KeepBindingsVisitor(builder::setBindings);
}
if (name.equals(Edge.preconditions)) {
return new KeepPreconditionsVisitor(builder::setPreconditions);
}
if (name.equals(Edge.consequences)) {
return new KeepConsequencesVisitor(builder::setConsequences);
}
return super.visitArray(name);
}
@Override
public void visitEnd() {
parent.accept(builder.setMetaInfo(metaInfoBuilder.build()).build());
}
}
private static class UsesReflectionVisitor extends AnnotationVisitorBase {
private final Parent<KeepEdge> parent;
private final KeepEdge.Builder builder = KeepEdge.builder();
private final KeepPreconditions.Builder preconditions = KeepPreconditions.builder();
private final KeepEdgeMetaInfo.Builder metaInfoBuilder = KeepEdgeMetaInfo.builder();
UsesReflectionVisitor(
Parent<KeepEdge> parent,
Consumer<KeepEdgeMetaInfo.Builder> addContext,
KeepItemPattern context) {
this.parent = parent;
preconditions.addCondition(KeepCondition.builder().setItemPattern(context).build());
addContext.accept(metaInfoBuilder);
}
@Override
public void visit(String name, Object value) {
if (name.equals(Edge.description) && value instanceof String) {
metaInfoBuilder.setDescription((String) value);
return;
}
super.visit(name, value);
}
@Override
public AnnotationVisitor visitArray(String name) {
if (name.equals(KeepConstants.UsesReflection.value)) {
return new KeepConsequencesVisitor(builder::setConsequences);
}
if (name.equals(KeepConstants.UsesReflection.additionalPreconditions)) {
return new KeepPreconditionsVisitor(
additionalPreconditions -> {
additionalPreconditions.forEach(preconditions::addCondition);
});
}
return super.visitArray(name);
}
@Override
public void visitEnd() {
parent.accept(
builder
.setMetaInfo(metaInfoBuilder.build())
.setPreconditions(preconditions.build())
.build());
}
}
private static class KeepBindingsVisitor extends AnnotationVisitorBase {
private final Parent<KeepBindings> parent;
private final KeepBindings.Builder builder = KeepBindings.builder();
public KeepBindingsVisitor(Parent<KeepBindings> parent) {
this.parent = parent;
}
@Override
public AnnotationVisitor visitAnnotation(String name, String descriptor) {
if (descriptor.equals(KeepConstants.Binding.DESCRIPTOR)) {
return new KeepBindingVisitor(builder);
}
return super.visitAnnotation(name, descriptor);
}
@Override
public void visitEnd() {
parent.accept(builder.build());
}
}
private static class KeepPreconditionsVisitor extends AnnotationVisitorBase {
private final Parent<KeepPreconditions> parent;
private final KeepPreconditions.Builder builder = KeepPreconditions.builder();
public KeepPreconditionsVisitor(Parent<KeepPreconditions> parent) {
this.parent = parent;
}
@Override
public AnnotationVisitor visitAnnotation(String name, String descriptor) {
if (descriptor.equals(Condition.DESCRIPTOR)) {
return new KeepConditionVisitor(builder::addCondition);
}
return super.visitAnnotation(name, descriptor);
}
@Override
public void visitEnd() {
parent.accept(builder.build());
}
}
private static class KeepConsequencesVisitor extends AnnotationVisitorBase {
private final Parent<KeepConsequences> parent;
private final KeepConsequences.Builder builder = KeepConsequences.builder();
public KeepConsequencesVisitor(Parent<KeepConsequences> parent) {
this.parent = parent;
}
@Override
public AnnotationVisitor visitAnnotation(String name, String descriptor) {
if (descriptor.equals(Target.DESCRIPTOR)) {
return KeepTargetVisitor.create(builder::addTarget);
}
return super.visitAnnotation(name, descriptor);
}
@Override
public void visitEnd() {
parent.accept(builder.build());
}
}
private abstract static class Declaration<T> {
abstract String kind();
abstract T getValue();
boolean tryParse(String name, Object value) {
return false;
}
AnnotationVisitor tryParseArray(String name, Consumer<T> onValue) {
return null;
}
}
private abstract static class SingleDeclaration<T> extends Declaration<T> {
private String declarationName = null;
private T declarationValue = null;
private AnnotationVisitor declarationVisitor = null;
abstract T getDefaultValue();
abstract T parse(String name, Object value);
AnnotationVisitor parseArray(String name, Consumer<T> setValue) {
return null;
}
private boolean hasDeclaration() {
return declarationValue != null || declarationVisitor != null;
}
private void error(String name) {
throw new KeepEdgeException(
"Multiple declarations defining "
+ kind()
+ ": '"
+ declarationName
+ "' and '"
+ name
+ "'");
}
@Override
public final T getValue() {
return declarationValue == null ? getDefaultValue() : declarationValue;
}
@Override
final boolean tryParse(String name, Object value) {
T result = parse(name, value);
if (result != null) {
if (hasDeclaration()) {
error(name);
}
declarationName = name;
declarationValue = result;
return true;
}
return false;
}
@Override
AnnotationVisitor tryParseArray(String name, Consumer<T> setValue) {
AnnotationVisitor visitor = parseArray(name, setValue.andThen(v -> declarationValue = v));
if (visitor != null) {
if (hasDeclaration()) {
error(name);
}
declarationName = name;
declarationVisitor = visitor;
return visitor;
}
return null;
}
}
private static class ClassDeclaration extends SingleDeclaration<KeepClassReference> {
@Override
String kind() {
return "class";
}
KeepClassReference wrap(KeepQualifiedClassNamePattern namePattern) {
return KeepClassReference.fromClassNamePattern(namePattern);
}
@Override
KeepClassReference getDefaultValue() {
return wrap(KeepQualifiedClassNamePattern.any());
}
@Override
KeepClassReference parse(String name, Object value) {
if (name.equals(Item.classFromBinding) && value instanceof String) {
return KeepClassReference.fromBindingReference((String) value);
}
if (name.equals(Item.classConstant) && value instanceof Type) {
return wrap(KeepQualifiedClassNamePattern.exact(((Type) value).getClassName()));
}
if (name.equals(Item.className) && value instanceof String) {
return wrap(KeepQualifiedClassNamePattern.exact(((String) value)));
}
return null;
}
}
private static class ExtendsDeclaration extends SingleDeclaration<KeepExtendsPattern> {
@Override
String kind() {
return "extends";
}
@Override
KeepExtendsPattern getDefaultValue() {
return KeepExtendsPattern.any();
}
@Override
KeepExtendsPattern parse(String name, Object value) {
if (name.equals(Item.extendsClassConstant) && value instanceof Type) {
return KeepExtendsPattern.builder()
.classPattern(KeepQualifiedClassNamePattern.exact(((Type) value).getClassName()))
.build();
}
if (name.equals(Item.extendsClassName) && value instanceof String) {
return KeepExtendsPattern.builder()
.classPattern(KeepQualifiedClassNamePattern.exact(((String) value)))
.build();
}
return null;
}
}
private static class MethodDeclaration extends Declaration<KeepMethodPattern> {
private KeepMethodPattern.Builder builder = null;
private KeepMethodPattern.Builder getBuilder() {
if (builder == null) {
builder = KeepMethodPattern.builder();
}
return builder;
}
@Override
String kind() {
return "method";
}
@Override
KeepMethodPattern getValue() {
return builder != null ? builder.build() : null;
}
@Override
boolean tryParse(String name, Object value) {
if (name.equals(Item.methodName) && value instanceof String) {
String methodName = (String) value;
if (!Item.methodNameDefaultValue.equals(methodName)) {
getBuilder().setNamePattern(KeepMethodNamePattern.exact(methodName));
}
return true;
}
if (name.equals(Item.methodReturnType) && value instanceof String) {
String returnType = (String) value;
if (!Item.methodReturnTypeDefaultValue.equals(returnType)) {
getBuilder()
.setReturnTypePattern(KeepEdgeReaderUtils.methodReturnTypeFromString(returnType));
}
return true;
}
return false;
}
@Override
AnnotationVisitor tryParseArray(String name, Consumer<KeepMethodPattern> ignored) {
if (name.equals(Item.methodParameters)) {
return new StringArrayVisitor(
params -> {
if (Arrays.asList(Item.methodParametersDefaultValue).equals(params)) {
return;
}
KeepMethodParametersPattern.Builder builder = KeepMethodParametersPattern.builder();
for (String param : params) {
builder.addParameterTypePattern(KeepEdgeReaderUtils.typePatternFromString(param));
}
KeepMethodParametersPattern result = builder.build();
getBuilder().setParametersPattern(result);
});
}
return null;
}
}
private static class FieldDeclaration extends Declaration<KeepFieldPattern> {
private KeepFieldPattern.Builder builder = null;
private KeepFieldPattern.Builder getBuilder() {
if (builder == null) {
builder = KeepFieldPattern.builder();
}
return builder;
}
@Override
String kind() {
return "field";
}
@Override
KeepFieldPattern getValue() {
return builder != null ? builder.build() : null;
}
@Override
boolean tryParse(String name, Object value) {
if (name.equals(Item.fieldName) && value instanceof String) {
String fieldName = (String) value;
if (!Item.fieldNameDefaultValue.equals(fieldName)) {
getBuilder().setNamePattern(KeepFieldNamePattern.exact(fieldName));
}
return true;
}
if (name.equals(Item.fieldType) && value instanceof String) {
String fieldType = (String) value;
if (!Item.fieldTypeDefaultValue.equals(fieldType)) {
getBuilder()
.setTypePattern(
KeepFieldTypePattern.fromType(
KeepEdgeReaderUtils.typePatternFromString(fieldType)));
}
return true;
}
return false;
}
}
private static class MemberDeclaration extends Declaration<KeepMemberPattern> {
private MethodDeclaration methodDeclaration = new MethodDeclaration();
private FieldDeclaration fieldDeclaration = new FieldDeclaration();
@Override
String kind() {
return "member";
}
@Override
public KeepMemberPattern getValue() {
KeepMethodPattern method = methodDeclaration.getValue();
KeepFieldPattern field = fieldDeclaration.getValue();
if (method != null && field != null) {
throw new KeepEdgeException("Cannot define both a field and a method pattern");
}
if (method != null) {
return method;
}
if (field != null) {
return field;
}
return KeepMemberPattern.none();
}
@Override
boolean tryParse(String name, Object value) {
return methodDeclaration.tryParse(name, value) || fieldDeclaration.tryParse(name, value);
}
@Override
AnnotationVisitor tryParseArray(String name, Consumer<KeepMemberPattern> ignored) {
AnnotationVisitor visitor = methodDeclaration.tryParseArray(name, v -> {});
if (visitor != null) {
return visitor;
}
return fieldDeclaration.tryParseArray(name, v -> {});
}
}
private abstract static class KeepItemVisitorBase extends AnnotationVisitorBase {
private Parent<KeepItemPattern> parent;
private final ClassDeclaration classDeclaration = new ClassDeclaration();
private final ExtendsDeclaration extendsDeclaration = new ExtendsDeclaration();
private final MemberDeclaration memberDeclaration = new MemberDeclaration();
public KeepItemVisitorBase(Parent<KeepItemPattern> parent) {
setParent(parent);
}
public KeepItemVisitorBase() {}
void setParent(Parent<KeepItemPattern> parent) {
assert parent != null;
assert this.parent == null;
this.parent = parent;
}
@Override
public void visit(String name, Object value) {
if (classDeclaration.tryParse(name, value)
|| extendsDeclaration.tryParse(name, value)
|| memberDeclaration.tryParse(name, value)) {
return;
}
super.visit(name, value);
}
@Override
public AnnotationVisitor visitArray(String name) {
AnnotationVisitor visitor = memberDeclaration.tryParseArray(name, v -> {});
if (visitor != null) {
return visitor;
}
return super.visitArray(name);
}
@Override
public void visitEnd() {
parent.accept(
KeepItemPattern.builder()
.setClassReference(classDeclaration.getValue())
.setExtendsPattern(extendsDeclaration.getValue())
.setMemberPattern(memberDeclaration.getValue())
.build());
}
}
private static class KeepBindingVisitor extends KeepItemVisitorBase {
private final KeepBindings.Builder builder;
private String bindingName;
private KeepItemPattern item;
public KeepBindingVisitor(KeepBindings.Builder builder) {
this.builder = builder;
setParent(item -> this.item = item);
}
@Override
public void visit(String name, Object value) {
if (name.equals(Binding.bindingName) && value instanceof String) {
bindingName = (String) value;
return;
}
super.visit(name, value);
}
@Override
public void visitEnd() {
super.visitEnd();
builder.addBinding(bindingName, item);
}
}
private static class StringArrayVisitor extends AnnotationVisitorBase {
private final Consumer<List<String>> fn;
private final List<String> strings = new ArrayList<>();
public StringArrayVisitor(Consumer<List<String>> fn) {
this.fn = fn;
}
@Override
public void visit(String name, Object value) {
if (value instanceof String) {
strings.add((String) value);
} else {
super.visit(name, value);
}
}
@Override
public void visitEnd() {
super.visitEnd();
fn.accept(strings);
}
}
private static class OptionsDeclaration extends SingleDeclaration<KeepOptions> {
@Override
String kind() {
return "options";
}
@Override
KeepOptions getDefaultValue() {
return KeepOptions.keepAll();
}
@Override
KeepOptions parse(String name, Object value) {
return null;
}
@Override
AnnotationVisitor parseArray(String name, Consumer<KeepOptions> setValue) {
if (name.equals(KeepConstants.Target.disallow)) {
return new KeepOptionsVisitor(
options -> setValue.accept(KeepOptions.disallowBuilder().addAll(options).build()));
}
if (name.equals(KeepConstants.Target.allow)) {
return new KeepOptionsVisitor(
options -> setValue.accept(KeepOptions.allowBuilder().addAll(options).build()));
}
return null;
}
}
private static class KeepTargetVisitor extends KeepItemVisitorBase {
private final KeepTarget.Builder builder;
private final OptionsDeclaration optionsDeclaration = new OptionsDeclaration();
static KeepTargetVisitor create(Parent<KeepTarget> parent) {
KeepTarget.Builder builder = KeepTarget.builder();
return new KeepTargetVisitor(parent, builder);
}
private KeepTargetVisitor(Parent<KeepTarget> parent, KeepTarget.Builder builder) {
super(item -> parent.accept(builder.setItemPattern(item).build()));
this.builder = builder;
}
@Override
public AnnotationVisitor visitArray(String name) {
AnnotationVisitor visitor = optionsDeclaration.tryParseArray(name, builder::setOptions);
if (visitor != null) {
return visitor;
}
return super.visitArray(name);
}
}
private static class KeepConditionVisitor extends KeepItemVisitorBase {
public KeepConditionVisitor(Parent<KeepCondition> parent) {
super(item -> parent.accept(KeepCondition.builder().setItemPattern(item).build()));
}
}
private static class KeepOptionsVisitor extends AnnotationVisitorBase {
private final Parent<Collection<KeepOption>> parent;
private final Set<KeepOption> options = new HashSet<>();
public KeepOptionsVisitor(Parent<Collection<KeepOption>> parent) {
this.parent = parent;
}
@Override
public void visitEnum(String ignore, String descriptor, String value) {
if (!descriptor.equals(KeepConstants.Option.DESCRIPTOR)) {
super.visitEnum(ignore, descriptor, value);
}
KeepOption option;
switch (value) {
case Option.SHRINKING:
option = KeepOption.SHRINKING;
break;
case Option.OPTIMIZATION:
option = KeepOption.OPTIMIZING;
break;
case Option.OBFUSCATION:
option = KeepOption.OBFUSCATING;
break;
case Option.ACCESS_MODIFICATION:
option = KeepOption.ACCESS_MODIFICATION;
break;
case Option.ANNOTATION_REMOVAL:
option = KeepOption.ANNOTATION_REMOVAL;
break;
default:
super.visitEnum(ignore, descriptor, value);
return;
}
options.add(option);
}
@Override
public void visitEnd() {
parent.accept(options);
super.visitEnd();
}
}
}