org.frameworkset.util.LocalVariableTableParameterNameDiscoverer Maven / Gradle / Ivy
/*
* Copyright 2008 biaoping.yin
*
* Licensed 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.frameworkset.util;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.HashMap;
import java.util.Map;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
//import bboss.org.objectweb.asm.ClassAdapter;
import bboss.org.objectweb.asm.ClassReader;
import bboss.org.objectweb.asm.ClassVisitor;
import bboss.org.objectweb.asm.Label;
import bboss.org.objectweb.asm.MethodVisitor;
import bboss.org.objectweb.asm.Opcodes;
import bboss.org.objectweb.asm.Type;
/**
* Title: LocalVariableTableParameterNameDiscoverer.java
* Description:
* bboss workgroup
* Copyright (c) 2007
* @Date 2010-10-23 下午10:41:14
* @author biaoping.yin
* @version 1.0
*/
public class LocalVariableTableParameterNameDiscoverer implements ParameterNameDiscoverer {
private static Logger logger = LoggerFactory.getLogger(LocalVariableTableParameterNameDiscoverer.class);
// private final Map parameterNamesCache =new java.util.concurrent.ConcurrentHashMap(16);
private final Map classReaderCache = new HashMap();
// public String[] getCachedParameterNames(Method method) {
// String[] paramNames = (String[]) this.parameterNamesCache.get(method);
//
// if (paramNames == null) {
// try {
// paramNames = visitMethod(method).getParameterNames();
// if (paramNames != null) {
// this.parameterNamesCache.put(method, paramNames);
// }
// }
// catch (IOException ex) {
// // We couldn't load the class file, which is not fatal as it
// // simply means this method of discovering parameter names won't work.
//
// logger.debug("IOException whilst attempting to read '.class' file for class [" +
// method.getDeclaringClass().getName() +
// "] - unable to determine parameter names for method: " + method, ex);
//
// }
// }
// return paramNames;
// }
public String[] getParameterNames(Method method) {
try {
String[] paramNames = visitMethod(method).getParameterNames();
return paramNames;
}
catch (IOException ex) {
// We couldn't load the class file, which is not fatal as it
// simply means this method of discovering parameter names won't work.
logger.error("IOException whilst attempting to read '.class' file for class [" +
method.getDeclaringClass().getName() +
"] - unable to determine parameter names for method: " + method, ex);
}
return null;
}
// public String[] getParameterNames(Constructor ctor) {
// String[] paramNames = (String[]) this.parameterNamesCache.get(ctor);
// if (paramNames == null) {
// try {
// paramNames = visitConstructor(ctor).getParameterNames();
// if (paramNames != null) {
// this.parameterNamesCache.put(ctor, paramNames);
// }
// }
// catch (IOException ex) {
// // We couldn't load the class file, which is not fatal as it
// // simply means this method of discovering parameter names won't work.
// if (logger.isDebugEnabled()) {
// logger.debug("IOException whilst attempting to read '.class' file for class [" +
// ctor.getDeclaringClass().getName() +
// "] - unable to determine parameter names for constructor: " + ctor, ex);
// }
// }
// }
// return paramNames;
// }
public String[] getParameterNames(Constructor ctor) {
// String[] paramNames = (String[]) this.parameterNamesCache.get(ctor);
// if (paramNames == null) {
try {
String[] paramNames = visitConstructor(ctor).getParameterNames();
return paramNames;
}
catch (IOException ex) {
// We couldn't load the class file, which is not fatal as it
// simply means this method of discovering parameter names won't work.
{
logger.error("IOException whilst attempting to read '.class' file for class [" +
ctor.getDeclaringClass().getName() +
"] - unable to determine parameter names for constructor: " + ctor, ex);
}
}
return null;
}
/**
* Visit the given method and discover its parameter names.
*/
private ParameterNameDiscoveringVisitor visitMethod(Method method) throws IOException {
ClassReader classReader = getClassReader(method.getDeclaringClass());
FindMethodParameterNamesClassVisitor classVisitor = new FindMethodParameterNamesClassVisitor(method);
//classReader.accept(classVisitor, false);
classReader.accept(classVisitor, 0);
return classVisitor;
}
/**
* Visit the given constructor and discover its parameter names.
*/
private ParameterNameDiscoveringVisitor visitConstructor(Constructor ctor) throws IOException {
ClassReader classReader = getClassReader(ctor.getDeclaringClass());
FindConstructorParameterNamesClassVisitor classVisitor = new FindConstructorParameterNamesClassVisitor(ctor);
classReader.accept(classVisitor, 0);
//classReader.accept(classVisitor, false);
return classVisitor;
}
/**
* Obtain a (cached) ClassReader for the given class.
*/
private ClassReader getClassReader(Class clazz) throws IOException {
clazz = ClassUtil.getClassInfo(clazz).getClazz();
ClassReader classReader = (ClassReader) this.classReaderCache.get(clazz);
if(classReader != null)
return classReader;
synchronized (this.classReaderCache) {
classReader = (ClassReader) this.classReaderCache.get(clazz);
if(classReader != null)
return classReader;
InputStream is = clazz.getResourceAsStream(ClassUtils.getClassFileName(clazz));
if (is == null) {
throw new FileNotFoundException("Class file for class [" + clazz.getName() + "] not found");
}
try {
classReader = new ClassReader(is);
this.classReaderCache.put(clazz, classReader);
}
finally {
is.close();
}
return classReader;
}
}
/**
* Helper class that looks for a given member name and descriptor, and then
* attempts to find the parameter names for that member.
*/
private static abstract class ParameterNameDiscoveringVisitor extends ClassVisitor {
private String methodNameToMatch;
private String descriptorToMatch;
private int numParamsExpected;
/*
* The nth entry contains the slot index of the LVT table entry holding the
* argument name for the nth parameter.
*/
private int[] lvtSlotIndex;
private String[] parameterNames;
public ParameterNameDiscoveringVisitor(String name, boolean isStatic, Class[] paramTypes) {
super(Opcodes.ASM5);
this.methodNameToMatch = name;
this.numParamsExpected = paramTypes.length;
computeLvtSlotIndices(isStatic, paramTypes);
}
public void setDescriptorToMatch(String descriptor) {
this.descriptorToMatch = descriptor;
}
public MethodVisitor visitMethod(int access, String name, String desc, String signature, String[] exceptions) {
if (name.equals(this.methodNameToMatch) && desc.equals(this.descriptorToMatch)) {
return new LocalVariableTableVisitor(this, isStatic(access));
}
else {
// Not interested in this method...
return null;
}
}
private boolean isStatic(int access) {
return ((access & Opcodes.ACC_STATIC) > 0);
}
public String[] getParameterNames() {
return this.parameterNames;
}
private void computeLvtSlotIndices(boolean isStatic, Class[] paramTypes) {
this.lvtSlotIndex = new int[paramTypes.length];
int nextIndex = (isStatic ? 0 : 1);
for (int i = 0; i < paramTypes.length; i++) {
this.lvtSlotIndex[i] = nextIndex;
if (isWideType(paramTypes[i])) {
nextIndex += 2;
}
else {
nextIndex++;
}
}
}
private boolean isWideType(Class aType) {
return (aType == Long.TYPE || aType == Double.TYPE);
}
}
private static class FindMethodParameterNamesClassVisitor extends ParameterNameDiscoveringVisitor {
public FindMethodParameterNamesClassVisitor(Method method) {
super(method.getName(), Modifier.isStatic(method.getModifiers()), method.getParameterTypes());
setDescriptorToMatch(Type.getMethodDescriptor(method));
}
}
private static class FindConstructorParameterNamesClassVisitor extends ParameterNameDiscoveringVisitor {
public FindConstructorParameterNamesClassVisitor(Constructor ctor) {
super("", false, ctor.getParameterTypes());
Type[] pTypes = new Type[ctor.getParameterTypes().length];
for (int i = 0; i < pTypes.length; i++) {
pTypes[i] = Type.getType(ctor.getParameterTypes()[i]);
}
setDescriptorToMatch(Type.getMethodDescriptor(Type.VOID_TYPE, pTypes));
}
}
private static class LocalVariableTableVisitor extends MethodVisitor {
private final ParameterNameDiscoveringVisitor memberVisitor;
private final boolean isStatic;
private String[] parameterNames;
private boolean hasLvtInfo = false;
public LocalVariableTableVisitor(ParameterNameDiscoveringVisitor memberVisitor, boolean isStatic) {
super(Opcodes.ASM5);
this.memberVisitor = memberVisitor;
this.isStatic = isStatic;
this.parameterNames = new String[memberVisitor.numParamsExpected];
}
public void visitLocalVariable(
String name, String description, String signature, Label start, Label end, int index) {
this.hasLvtInfo = true;
int[] lvtSlotIndices = this.memberVisitor.lvtSlotIndex;
for (int i = 0; i < lvtSlotIndices.length; i++) {
if (lvtSlotIndices[i] == index) {
this.parameterNames[i] = name;
}
}
}
public void visitEnd() {
if (this.hasLvtInfo || (this.isStatic && this.parameterNames.length == 0)) {
// visitLocalVariable will never be called for static no args methods
// which doesn't use any local variables.
// This means that hasLvtInfo could be false for that kind of methods
// even if the class has local variable info.
this.memberVisitor.parameterNames = this.parameterNames;
}
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy