oracle.toplink.essentials.internal.weaving.TopLinkMethodWeaver Maven / Gradle / Ivy
The newest version!
/*
* The contents of this file are subject to the terms
* of the Common Development and Distribution License
* (the "License"). You may not use this file except
* in compliance with the License.
*
* You can obtain a copy of the license at
* glassfish/bootstrap/legal/CDDLv1.0.txt or
* https://glassfish.dev.java.net/public/CDDLv1.0.html.
* See the License for the specific language governing
* permissions and limitations under the License.
*
* When distributing Covered Code, include this CDDL
* HEADER in each file and include the License file at
* glassfish/bootstrap/legal/CDDLv1.0.txt. If applicable,
* add the following below this CDDL HEADER, with the
* fields enclosed by brackets "[]" replaced with your
* own identifying information: Portions Copyright [yyyy]
* [name of copyright owner]
*/
// Copyright (c) 2005, 2006, Oracle. All rights reserved.
package oracle.toplink.essentials.internal.weaving;
//ASM imports
import oracle.toplink.libraries.asm.*;
import java.util.Iterator;
/**
* INTERNAL:
*
* Used by TopLink's weaving feature to adjust methods to make use of ValueHolders that
* have been inserted by TopLinkClassWeaver.
*
* For FIELD access, changes references to GETFIELD and PUTFIELD to call newly added
* convenience methods.
*
* For Property access, modifies the getters and setters to make use of new ValueHolders
*
* Also adds initialization of newly added ValueHolders to constructor.
*
*/
public class TopLinkMethodWeaver extends CodeAdapter implements Constants {
protected TopLinkClassWeaver tcw;
protected String methodName;
private String methodDescriptor = null;
// used to determine if we are at the first line of a method
private boolean methodStarted = false;
// Used to control initialization of valueholders in constructor
private boolean constructorInitializationDone = false;
public TopLinkMethodWeaver(TopLinkClassWeaver tcw, String methodName, String methodEscriptor,
CodeVisitor cv) {
super(cv);
this.tcw = tcw;
this.methodName = methodName;
this.methodDescriptor = methodDescriptor;
}
/**
* INTERNAL:
* Change GETFIELD and PUTFIELD for fields that use attribute access to make use of new convenience methods
*
* A GETFIELD for an attribute named 'variableName' will be replaced by a call to:
*
* _toplink_getvariableName()
*
* A PUTFIELD for an attribute named 'variableName' will be replaced by a call to:
*
* toplink_setvariableName(variableName)
*/
public void weaveAttributesIfRequired(int opcode, String owner, String name, String desc){
AttributeDetails attributeDetails = (AttributeDetails)tcw.classDetails.getAttributeDetailsFromClassOrSuperClass(name);
if (attributeDetails == null || !attributeDetails.isMappedWithAttributeAccess()){
super.visitFieldInsn(opcode, owner, name, desc);
return;
}
if (opcode == GETFIELD) {
if (attributeDetails != null) {
cv.visitMethodInsn(INVOKEVIRTUAL, tcw.classDetails.getClassName(), "_toplink_get" + name, "()L" + attributeDetails.getReferenceClass().replace('.','/') + ";");
} else {
super.visitFieldInsn(opcode, owner, name, desc);
}
} else if (opcode == PUTFIELD) {
if (attributeDetails != null) {
cv.visitMethodInsn(INVOKEVIRTUAL, tcw.classDetails.getClassName(), "_toplink_set" + name, "(L" + attributeDetails.getReferenceClass().replace('.','/') + ";)V");
} else {
super.visitFieldInsn(opcode, owner, name, desc);
}
} else {
super.visitFieldInsn(opcode, owner, name, desc);
}
}
/**
* INTERNAL:
* Add initialization of new ValueHolders to constuctors. If a ValueHolder called 'variableName'
* has been added, the following line will be added to the constructor.
*
* _toplink_variableName_vh = new ValueHolder();
*/
public void weaveConstructorIfRequired(int opcode, String owner, String name, String desc){
if (!constructorInitializationDone && ("".equals(methodName)||"".equals(methodName))) {
// look for the superclass initializer and insert the valueholder
// initialization after it
if (opcode == INVOKESPECIAL && name.startsWith("")) {
ClassDetails details = tcw.classDetails;
Iterator attributes = details.getAttributesMap().keySet().iterator();
while (attributes.hasNext()){
String key = (String)attributes.next();
AttributeDetails attribute = (AttributeDetails)details.getAttributesMap().get(key);
if (attribute.weaveValueHolders() && !attribute.isCollectionMapping() && !attribute.isAttributeOnSuperClass()){
super.visitVarInsn(ALOAD, 0);
super.visitTypeInsn(NEW, "oracle/toplink/essentials/indirection/ValueHolder");
super.visitInsn(DUP);
super.visitMethodInsn(INVOKESPECIAL, "oracle/toplink/essentials/indirection/ValueHolder", "", "()V");
super.visitFieldInsn(PUTFIELD, details.className, "_toplink_" + attribute.attributeName + "_vh", "Loracle/toplink/essentials/indirection/ValueHolderInterface;");
}
}
}
constructorInitializationDone = true;
}
}
/**
* INTERNAL:
* Modifies getter and setter methods for attributes using property access
*
* In a getter method for 'attributeName', the following line is added at the beginning of the method
*
* setAttributeName((<AttributeClass>)_toplink_attributeName_vh.getValue());
*
* In a setter method, for 'attributeName', the following line is added at the beginning of the method
*
* _toplink_attributeName_vh.setValue(methodArgument);
*
* TODO: In the end, the call to setValue() should be modified to somehow make use of the result of
* the getter method. This behavior has not yet been implemented.
*/
public void addValueHolderReferencesIfRequired(){
if (methodStarted){
return;
}
AttributeDetails attributeDetails = (AttributeDetails)tcw.classDetails.getGetterMethodToAttributeDetails().get(methodName);
if (attributeDetails != null && !attributeDetails.isAttributeOnSuperClass()){
cv.visitVarInsn(ALOAD, 0);
cv.visitVarInsn(ALOAD, 0);
cv.visitFieldInsn(GETFIELD, tcw.classDetails.getClassName(), "_toplink_" + attributeDetails.getAttributeName() + "_vh", TopLinkClassWeaver.VHI_SIGNATURE);
cv.visitMethodInsn(INVOKEINTERFACE, TopLinkClassWeaver.VHI_SHORT_SIGNATURE, "getValue", "()Ljava/lang/Object;");
cv.visitTypeInsn(CHECKCAST, attributeDetails.getReferenceClass().replace('.','/'));
cv.visitMethodInsn(INVOKEVIRTUAL, tcw.classDetails.getClassName(), attributeDetails.getSetterMethodName(), "(L" + attributeDetails.getReferenceClass().replace('.','/') + ";)V");
} else {
attributeDetails = (AttributeDetails)tcw.classDetails.getSetterMethodToAttributeDetails().get(methodName);
if (attributeDetails != null){
cv.visitVarInsn(ALOAD, 0);
cv.visitFieldInsn(GETFIELD, tcw.classDetails.getClassName(), "_toplink_" + attributeDetails.getAttributeName() + "_vh", "Loracle/toplink/essentials/indirection/ValueHolderInterface;");
cv.visitVarInsn(ALOAD, 1);
cv.visitMethodInsn(INVOKEINTERFACE, TopLinkClassWeaver.VHI_SHORT_SIGNATURE, "setValue", "(Ljava/lang/Object;)V");
}
}
}
public void visitInsn (final int opcode) {
addValueHolderReferencesIfRequired();
methodStarted = true;
super.visitInsn(opcode);
}
public void visitIntInsn (final int opcode, final int operand) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitIntInsn(opcode, operand);
}
public void visitVarInsn (final int opcode, final int var) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitVarInsn(opcode, var);
}
public void visitTypeInsn (final int opcode, final String desc) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitTypeInsn(opcode, desc);
}
public void visitFieldInsn (final int opcode, final String owner, final String name, final String desc){
addValueHolderReferencesIfRequired();
methodStarted = true;
weaveAttributesIfRequired(opcode, owner, name, desc);
}
public void visitMethodInsn (final int opcode, final String owner, final String name, final String desc){
addValueHolderReferencesIfRequired();
methodStarted = true;
super.visitMethodInsn(opcode, owner, name, desc);
weaveConstructorIfRequired(opcode, owner, name, desc);
}
public void visitJumpInsn (final int opcode, final Label label) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitJumpInsn(opcode, label);
}
public void visitLabel (final Label label) {
cv.visitLabel(label);
}
public void visitLdcInsn (final Object cst) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitLdcInsn(cst);
}
public void visitIincInsn (final int var, final int increment) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitIincInsn(var, increment);
}
public void visitTableSwitchInsn (final int min, final int max, final Label dflt, final Label labels[]){
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitTableSwitchInsn(min, max, dflt, labels);
}
public void visitLookupSwitchInsn (final Label dflt, final int keys[], final Label labels[]){
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitLookupSwitchInsn(dflt, keys, labels);
}
public void visitMultiANewArrayInsn (final String desc, final int dims) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitMultiANewArrayInsn(desc, dims);
}
public void visitTryCatchBlock (final Label start, final Label end,final Label handler, final String type){
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitTryCatchBlock(start, end, handler, type);
}
public void visitMaxs (final int maxStack, final int maxLocals) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitMaxs(0, 0);
}
public void visitLocalVariable (final String name, final String desc, final Label start, final Label end, final int index){
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitLocalVariable(name, desc, start, end, index);
}
public void visitLineNumber (final int line, final Label start) {
cv.visitLineNumber(line, start);
}
public void visitAttribute (final Attribute attr) {
addValueHolderReferencesIfRequired();
methodStarted = true;
cv.visitAttribute(attr);
}
// helper methods
protected AttributeDetails weaveValueHolders(ClassDetails startingDetails,
String fieldName) {
if (startingDetails == null) {
return null;
} else {
AttributeDetails attributeDetails = (AttributeDetails)startingDetails.getAttributesMap().get(fieldName);
if (attributeDetails != null && attributeDetails.weaveValueHolders()) {
return attributeDetails;
} else {
return weaveValueHolders(startingDetails.getSuperClassDetails(),
fieldName);
}
}
}
}