Package org.codehaus.aspectwerkz.transform.inlining.weaver

Source Code of org.codehaus.aspectwerkz.transform.inlining.weaver.AddMixinMethodsVisitor$AppendToInitMethodCodeAdapter

/**************************************************************************************
* Copyright (c) Jonas Bon�r, Alexandre Vasseur. All rights reserved.                 *
* http://aspectwerkz.codehaus.org                                                    *
* ---------------------------------------------------------------------------------- *
* The software in this package is published under the terms of the LGPL license      *
* a copy of which has been included with this distribution in the license.txt file.  *
**************************************************************************************/
package org.codehaus.aspectwerkz.transform.inlining.weaver;

import java.util.Iterator;
import java.util.Set;
import java.util.List;
import java.util.Map;
import java.util.HashMap;
import java.util.HashSet;

import org.objectweb.asm.*;
import org.codehaus.aspectwerkz.transform.Context;
import org.codehaus.aspectwerkz.transform.TransformationConstants;
import org.codehaus.aspectwerkz.transform.TransformationUtil;
import org.codehaus.aspectwerkz.transform.inlining.ContextImpl;
import org.codehaus.aspectwerkz.transform.inlining.AsmHelper;
import org.codehaus.aspectwerkz.definition.SystemDefinition;
import org.codehaus.aspectwerkz.definition.MixinDefinition;
import org.codehaus.aspectwerkz.expression.ExpressionContext;
import org.codehaus.aspectwerkz.expression.PointcutType;
import org.codehaus.aspectwerkz.reflect.ClassInfo;
import org.codehaus.aspectwerkz.reflect.MethodInfo;
import org.codehaus.aspectwerkz.reflect.ClassInfoHelper;
import org.codehaus.aspectwerkz.reflect.FieldInfo;
import org.codehaus.aspectwerkz.DeploymentModel;
import org.codehaus.aspectwerkz.DeploymentModel;
import org.codehaus.aspectwerkz.exception.DefinitionException;

/**
* Adds mixin methods and fields to hold mixin instances to the target class.
*
* @author <a href="mailto:jboner@codehaus.org">Jonas Bon�r </a>
*/
public class AddMixinMethodsVisitor extends ClassAdapter implements TransformationConstants {

    private final ContextImpl m_ctx;
    private String m_declaringTypeName;
    private final ClassInfo m_classInfo;
    private final Set m_addedMethods;
    private ExpressionContext m_expressionContext;
    private boolean m_hasClinit = false;
    private Map m_mixinFields;
    private boolean m_isAdvised = false;

    /**
     * Creates a new class adapter.
     *
     * @param cv
     * @param classInfo
     * @param ctx
     * @param addedMethods
     */
    public AddMixinMethodsVisitor(final ClassVisitor cv,
                                  final ClassInfo classInfo,
                                  final Context ctx,
                                  final Set addedMethods) {
        super(cv);
        m_classInfo = classInfo;
        m_ctx = (ContextImpl) ctx;
        m_addedMethods = addedMethods;
        m_expressionContext = new ExpressionContext(PointcutType.WITHIN, m_classInfo, m_classInfo);
    }

    /**
     * Visits the class.
     *
     * @param access
     * @param name
     * @param superName
     * @param interfaces
     * @param sourceFile
     */
    public void visit(final int version,
                      final int access,
                      final String name,
                      final String superName,
                      final String[] interfaces,
                      final String sourceFile) {
        ExpressionContext ctx = new ExpressionContext(PointcutType.WITHIN, m_classInfo, m_classInfo);
        if (!classFilter(m_classInfo, ctx, m_ctx.getDefinitions())) {
            m_declaringTypeName = name;
            m_mixinFields = new HashMap();

            // populate with fields already present for mixins from previous weaving
            for (int i = 0; i < m_classInfo.getFields().length; i++) {
                FieldInfo fieldInfo = m_classInfo.getFields()[i];
                if (fieldInfo.getName().startsWith(MIXIN_FIELD_NAME)) {
                    m_mixinFields.put(fieldInfo.getType(), fieldInfo);
                }
            }

            // add fields and method for (not already there) mixins
            addMixinMembers();
        }
        super.visit(version, access, name, superName, interfaces, sourceFile);
    }

    /**
     * Adds mixin fields and methods to the target class.
     */
    private void addMixinMembers() {
        int index = 0;
        for (Iterator it = m_ctx.getDefinitions().iterator(); it.hasNext();) {
            List mixinDefs = ((SystemDefinition) it.next()).getMixinDefinitions(m_expressionContext);

            // check for method clashes
            Set interfaceSet = new HashSet();
            for (Iterator it2 = mixinDefs.iterator(); it2.hasNext();) {
                interfaceSet.addAll(((MixinDefinition) it2.next()).getInterfaceClassNames());
            }
            if (ClassInfoHelper.hasMethodClash(interfaceSet, m_ctx.getLoader())) {
                return;
            }

            for (Iterator it2 = mixinDefs.iterator(); it2.hasNext();) {
                final MixinDefinition mixinDef = (MixinDefinition) it2.next();
                final ClassInfo mixinImpl = mixinDef.getMixinImpl();
                final DeploymentModel deploymentModel = mixinDef.getDeploymentModel();

                if (m_mixinFields.containsKey(mixinImpl)) {
                    continue;
                }
                final MixinFieldInfo fieldInfo = new MixinFieldInfo();
                fieldInfo.fieldName = MIXIN_FIELD_NAME + index;
                fieldInfo.mixinClassInfo = mixinImpl;

                addMixinField(fieldInfo, deploymentModel, mixinDef);
                addMixinMethods(fieldInfo, mixinDef);

                index++;
                m_isAdvised = true;
            }
        }
    }

    /**
     * Appends mixin instantiation to the clinit method and/or init method.
     *
     * @param access
     * @param name
     * @param desc
     * @param exceptions
     * @param attrs
     * @return
     */
    public CodeVisitor visitMethod(final int access,
                                   final String name,
                                   final String desc,
                                   final String[] exceptions,
                                   final Attribute attrs) {
        if (m_isAdvised) {
            if (name.equals(CLINIT_METHOD_NAME)) {
                m_hasClinit = true;
                CodeVisitor mv = new PrependToClinitMethodCodeAdapter(
                        cv.visitMethod(access, name, desc, exceptions, attrs)
                );
                mv.visitMaxs(0, 0);
                return mv;
            } else if (name.equals(INIT_METHOD_NAME)) {
                CodeVisitor mv = new AppendToInitMethodCodeAdapter(
                        cv.visitMethod(access, name, desc, exceptions, attrs)
                );
                mv.visitMaxs(0, 0);
                return mv;
            }
        }
        return super.visitMethod(access, name, desc, exceptions, attrs);
    }

    /**
     * Creates a new clinit method and adds mixin instantiation if it does not exist.
     */
    public void visitEnd() {
        if (m_isAdvised && !m_hasClinit) {
            // add the <clinit> method
            CodeVisitor mv = cv.visitMethod(
                    ACC_STATIC, CLINIT_METHOD_NAME, NO_PARAM_RETURN_VOID_SIGNATURE, null, null
            );
            for (Iterator i4 = m_mixinFields.values().iterator(); i4.hasNext();) {
                MixinFieldInfo fieldInfo = (MixinFieldInfo) i4.next();
                if (fieldInfo.isStatic) {
                    initializeStaticMixinField(mv, fieldInfo);
                }
            }

            mv.visitInsn(RETURN);
            mv.visitMaxs(0, 0);
        }
        super.visitEnd();
    }

    /**
     * Initializes a static mixin field.
     *
     * @param mv
     * @param fieldInfo
     */
    private void initializeStaticMixinField(final CodeVisitor mv, final MixinFieldInfo fieldInfo) {
        mv.visitLdcInsn(fieldInfo.mixinClassInfo.getName().replace('/', '.'));
        if (fieldInfo.isPerJVM) {
            mv.visitFieldInsn(
                    GETSTATIC,
                    m_declaringTypeName,
                    TARGET_CLASS_FIELD_NAME,
                    CLASS_CLASS_SIGNATURE
            );
            mv.visitMethodInsn(
                    INVOKEVIRTUAL,
                    CLASS_CLASS,
                    GETCLASSLOADER_METHOD_NAME,
                    CLASS_CLASS_GETCLASSLOADER_METHOD_SIGNATURE
            );
            mv.visitMethodInsn(
                    INVOKESTATIC,
                    MIXINS_CLASS_NAME,
                    MIXIN_OF_METHOD_NAME,
                    MIXIN_OF_METHOD_PER_JVM_SIGNATURE
            );
        } else {
            mv.visitFieldInsn(
                    GETSTATIC,
                    m_declaringTypeName,
                    TARGET_CLASS_FIELD_NAME,
                    CLASS_CLASS_SIGNATURE
            );
            mv.visitMethodInsn(
                    INVOKESTATIC,
                    MIXINS_CLASS_NAME,
                    MIXIN_OF_METHOD_NAME,
                    MIXIN_OF_METHOD_PER_CLASS_SIGNATURE
            );
        }
        mv.visitTypeInsn(CHECKCAST, fieldInfo.mixinClassInfo.getName().replace('.', '/'));
        mv.visitFieldInsn(
                PUTSTATIC,
                m_declaringTypeName,
                fieldInfo.fieldName,
                fieldInfo.mixinClassInfo.getSignature()
        );
    }

    /**
     * Initializes a member mixin field.
     *
     * @param mv
     * @param fieldInfo
     */
    private void initializeMemberMixinField(final CodeVisitor mv, final MixinFieldInfo fieldInfo) {
        mv.visitVarInsn(ALOAD, 0);
        mv.visitLdcInsn(fieldInfo.mixinClassInfo.getName().replace('/', '.'));
        mv.visitVarInsn(ALOAD, 0);
        mv.visitMethodInsn(
                INVOKESTATIC,
                MIXINS_CLASS_NAME,
                MIXIN_OF_METHOD_NAME,
                MIXIN_OF_METHOD_PER_INSTANCE_SIGNATURE
        );
        mv.visitTypeInsn(CHECKCAST, fieldInfo.mixinClassInfo.getName().replace('.', '/'));
        mv.visitFieldInsn(
                PUTFIELD,
                m_declaringTypeName,
                fieldInfo.fieldName,
                fieldInfo.mixinClassInfo.getSignature()
        );
    }

    /**
     * Adds the mixin field to the target class.
     *
     * @param fieldInfo
     * @param deploymentModel
     * @param mixinDef
     */
    private void addMixinField(final MixinFieldInfo fieldInfo,
                               final DeploymentModel deploymentModel,
                               final MixinDefinition mixinDef) {
        final String signature = fieldInfo.mixinClassInfo.getSignature();
        int modifiers = 0;
        if (deploymentModel.equals(DeploymentModel.PER_CLASS) || deploymentModel.equals(DeploymentModel.PER_JVM)) {
            fieldInfo.isStatic = true;
            fieldInfo.isPerJVM = deploymentModel.equals(DeploymentModel.PER_JVM);
            modifiers = ACC_PRIVATE + ACC_FINAL + ACC_STATIC + ACC_SYNTHETIC;
        } else if (deploymentModel.equals(DeploymentModel.PER_INSTANCE)) {
            fieldInfo.isStatic = false;
            modifiers = ACC_PRIVATE + ACC_FINAL + ACC_SYNTHETIC;
        } else {
            throw new DefinitionException(
                    "deployment model [" + mixinDef.getDeploymentModel() +
                    "] for mixin [" + mixinDef.getMixinImpl().getName() +
                    "] is not supported"
            );

        }
        if (mixinDef.isTransient()) {
            modifiers += ACC_TRANSIENT;
        }
        cv.visitField(modifiers, fieldInfo.fieldName, signature, null, null);
        m_mixinFields.put(mixinDef.getMixinImpl(), fieldInfo);
    }

    /**
     * Adds the mixin methods to the target class.
     *
     * @param fieldInfo
     * @param mixinDef
     */
    private void addMixinMethods(final MixinFieldInfo fieldInfo, final MixinDefinition mixinDef) {
        for (Iterator it3 = mixinDef.getMethodsToIntroduce().iterator(); it3.hasNext();) {
            MethodInfo methodInfo = (MethodInfo) it3.next();
            final String methodName = methodInfo.getName();
            final String methodSignature = methodInfo.getSignature();

            if (m_addedMethods.contains(AlreadyAddedMethodVisitor.getMethodKey(methodName, methodSignature))) {
                continue;
            }

            CodeVisitor mv = cv.visitMethod(
                    ACC_PUBLIC + ACC_SYNTHETIC,
                    methodName,
                    methodSignature,
                    null,
                    null
            );
            if (fieldInfo.isStatic) {
                mv.visitFieldInsn(
                        GETSTATIC,
                        m_declaringTypeName,
                        fieldInfo.fieldName,
                        fieldInfo.mixinClassInfo.getSignature()
                );
            } else {
                mv.visitVarInsn(ALOAD, 0);
                mv.visitFieldInsn(
                        GETFIELD,
                        m_declaringTypeName,
                        fieldInfo.fieldName,
                        fieldInfo.mixinClassInfo.getSignature()
                );
            }
            AsmHelper.loadArgumentTypes(mv, Type.getArgumentTypes(methodSignature), false);
            mv.visitMethodInsn(
                    INVOKEVIRTUAL,
                    fieldInfo.mixinClassInfo.getName().replace('.', '/'),
                    methodName,
                    methodSignature
            );
            AsmHelper.addReturnStatement(mv, Type.getReturnType(methodSignature));
            mv.visitMaxs(0, 0);
        }
    }

    /**
     * Filters the classes to be transformed.
     *
     * @param classInfo   the class to filter
     * @param ctx         the context
     * @param definitions a set with the definitions
     * @return boolean true if the method should be filtered away
     */
    public static boolean classFilter(final ClassInfo classInfo,
                                      final ExpressionContext ctx,
                                      final Set definitions) {
        for (Iterator it = definitions.iterator(); it.hasNext();) {
            SystemDefinition systemDef = (SystemDefinition) it.next();
            if (classInfo.isInterface()) {
                return true;
            }
            String className = classInfo.getName().replace('/', '.');
            if (systemDef.inExcludePackage(className)) {
                return true;
            }
            if (!systemDef.inIncludePackage(className)) {
                return true;
            }
            if (systemDef.hasMixin(ctx)) {
                return false;
            }
        }
        return true;
    }

    /**
     * Adds initialization of static mixin fields to the beginning of the clinit method.
     *
     * @author <a href="mailto:jboner@codehaus.org">Jonas Bon�r </a>
     */
    public class PrependToClinitMethodCodeAdapter extends CodeAdapter {

        public PrependToClinitMethodCodeAdapter(final CodeVisitor ca) {
            super(ca);
            for (Iterator i4 = m_mixinFields.values().iterator(); i4.hasNext();) {
                MixinFieldInfo fieldInfo = (MixinFieldInfo) i4.next();
                if (fieldInfo.isStatic) {
                    initializeStaticMixinField(ca, fieldInfo);
                }
            }
        }
    }

    /**
     * Adds initialization of member mixin fields to end of the init method.
     *
     * @author <a href="mailto:jboner@codehaus.org">Jonas Bon�r </a>
     */
    public class AppendToInitMethodCodeAdapter extends CodeAdapter {

        public AppendToInitMethodCodeAdapter(final CodeVisitor ca) {
            super(ca);
        }

        public void visitInsn(final int opcode) {
            if (opcode == RETURN) {
                for (Iterator i4 = m_mixinFields.values().iterator(); i4.hasNext();) {
                    MixinFieldInfo fieldInfo = (MixinFieldInfo) i4.next();
                    if (!fieldInfo.isStatic) {
                        initializeMemberMixinField(cv, fieldInfo);
                    }
                }
            }
            super.visitInsn(opcode);
        }
    }

    private static class MixinFieldInfo {
        private String fieldName;
        private ClassInfo mixinClassInfo;
        private boolean isStatic;
        private boolean isPerJVM = false;
    }
}
TOP

Related Classes of org.codehaus.aspectwerkz.transform.inlining.weaver.AddMixinMethodsVisitor$AppendToInitMethodCodeAdapter

TOP
Copyright © 2018 www.massapi.com. All rights reserved.
All source code are property of their respective owners. Java is a trademark of Sun Microsystems, Inc and owned by ORACLE Inc. Contact coftware#gmail.com.