| /* |
| * Licensed to the Apache Software Foundation (ASF) under one |
| * or more contributor license agreements. See the NOTICE file |
| * distributed with this work for additional information |
| * regarding copyright ownership. The ASF licenses this file |
| * to you under the Apache License, Version 2.0 (the |
| * "License"); you may not use this file except in compliance |
| * with the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package org.apache.drill.exec.compile; |
| |
| import java.util.HashMap; |
| |
| import org.objectweb.asm.AnnotationVisitor; |
| import org.objectweb.asm.Attribute; |
| import org.objectweb.asm.ClassVisitor; |
| import org.objectweb.asm.FieldVisitor; |
| import org.objectweb.asm.MethodVisitor; |
| import org.objectweb.asm.TypePath; |
| |
| /** |
| * A ClassVisitor that verifies the required call sequence described in |
| * http://asm.ow2.org/asm50/javadoc/user/org/objectweb/asm/ClassVisitor.html . |
| * |
| * <p> |
| * There is no CheckAnnotationVisitorFsm at this time. There is no |
| * CheckFieldVisitorFsm at this time. |
| */ |
| public class CheckClassVisitorFsm extends ClassVisitor { |
| /* |
| * From |
| * http://asm.ow2.org/asm50/javadoc/user/org/objectweb/asm/ClassVisitor.html |
| * "A visitor to visit a Java class. The methods of this class must be called |
| * in the following order: visit [ visitSource ] [ visitOuterClass ] ( |
| * visitAnnotation | visitTypeAnnotation | visitAttribute )* ( visitInnerClass |
| * | visitField | visitMethod )* visitEnd." |
| */ |
| private final static FsmDescriptor fsmDescriptor = createFsmDescriptor(); |
| |
| private static FsmDescriptor createFsmDescriptor() { |
| final HashMap<String, Character> tokenMap = new HashMap<>(); |
| tokenMap.put("visit", 'v'); |
| tokenMap.put("visitSource", 'S'); |
| tokenMap.put("visitOuterClass", 'O'); |
| tokenMap.put("visitAnnotation", 'A'); |
| tokenMap.put("visitTypeAnnotation", 'T'); |
| tokenMap.put("visitAttribute", 'R'); |
| tokenMap.put("visitInnerClass", 'I'); |
| tokenMap.put("visitField", 'F'); |
| tokenMap.put("visitMethod", 'M'); |
| tokenMap.put("visitEnd", 'E'); |
| |
| return new FsmDescriptor(tokenMap, "vS?+O?+(A|T|R)*+(I|F|M)*+E", "visitEnd"); |
| } |
| |
| private final FsmCursor fsmCursor; |
| |
| /** |
| * See {@link org.objectweb.asm.ClassVisitor#ClassVisitor(int, ClassVisitor)}. |
| */ |
| public CheckClassVisitorFsm(final int api, final ClassVisitor cv) { |
| super(api, cv); |
| fsmCursor = fsmDescriptor.createCursor(); |
| } |
| |
| @Override |
| public void visit(final int version, final int access, final String name, |
| final String signature, final String superName, final String[] interfaces) { |
| fsmCursor.transition("visit"); |
| super.visit(version, access, name, signature, superName, interfaces); |
| } |
| |
| @Override |
| public AnnotationVisitor visitAnnotation(final String desc, |
| final boolean visible) { |
| fsmCursor.transition("visitAnnotation"); |
| final AnnotationVisitor annotationVisitor = super.visitAnnotation(desc, |
| visible); |
| return annotationVisitor; // TODO: add CheckAnnotationVisitorFsm |
| } |
| |
| @Override |
| public void visitAttribute(final Attribute attr) { |
| fsmCursor.transition("visitAttribute"); |
| super.visitAttribute(attr); |
| } |
| |
| @Override |
| public void visitEnd() { |
| fsmCursor.transition("visitEnd"); |
| fsmCursor.reset(); |
| super.visitEnd(); |
| } |
| |
| @Override |
| public FieldVisitor visitField(final int access, final String name, |
| final String desc, final String signature, final Object value) { |
| fsmCursor.transition("visitField"); |
| final FieldVisitor fieldVisitor = super.visitField(access, name, desc, |
| signature, value); |
| return fieldVisitor; // TODO: add CheckFieldVisitorFsm |
| } |
| |
| @Override |
| public void visitInnerClass(final String name, final String outerName, |
| final String innerName, final int access) { |
| fsmCursor.transition("visitInnerClass"); |
| super.visitInnerClass(name, outerName, innerName, access); |
| } |
| |
| @Override |
| public MethodVisitor visitMethod(final int access, final String name, |
| final String desc, final String signature, final String[] exceptions) { |
| fsmCursor.transition("visitMethod"); |
| final MethodVisitor methodVisitor = super.visitMethod(access, name, desc, |
| signature, exceptions); |
| return new CheckMethodVisitorFsm(api, methodVisitor); |
| } |
| |
| @Override |
| public void visitOuterClass(final String owner, final String name, |
| final String desc) { |
| fsmCursor.transition("visitOuterClass"); |
| super.visitOuterClass(owner, name, desc); |
| } |
| |
| @Override |
| public void visitSource(final String source, final String debug) { |
| fsmCursor.transition("visitSource"); |
| super.visitSource(source, debug); |
| } |
| |
| @Override |
| public AnnotationVisitor visitTypeAnnotation(final int typeRef, |
| final TypePath typePath, final String desc, final boolean visible) { |
| fsmCursor.transition("visitTypeAnnotation"); |
| final AnnotationVisitor annotationVisitor = super.visitTypeAnnotation( |
| typeRef, typePath, desc, visible); |
| return annotationVisitor; // TODO: add CheckAnnotationVisitorFsm |
| } |
| } |