All Downloads are FREE. Search and download functionalities are using the official Maven repository.

gnu.jel.Library Maven / Gradle / Ivy

There is a newer version: 0.8.1
Show newest version
/*-*- Mode: Java; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
 * $Id: Library.java 490 2006-10-01 16:08:04Z metlov $
 *
 * This file is part of the Java Expressions Library (JEL).
 *   For more information about JEL visit :
 *    http://kinetic.ac.donetsk.ua/JEL/
 *
 * (c) 1998 -- 2007 by Konstantin Metlov(metlov@kinetic.ac.donetsk.ua);
 *
 * JEL is Distributed under the terms of GNU General Public License.
 *    This code comes with ABSOLUTELY NO WARRANTY.
 *  For license details see COPYING file in this directory.
 */

package gnu.jel;

import gnu.jel.debug.Debug;
import java.lang.reflect.Member;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.lang.reflect.Constructor;
import java.util.HashMap;
import java.util.ArrayList;
import java.util.Iterator;

/**
 * A namespace for JEL expressions.
 * 

There are two types of members in the library, those which are stateless * (i.e. their value depends only on their arguments, if there are any) and * stateful (also called here dynamic). The result of evaluation of * stateful members may depend on other factors besides their arguments. * *

Examples of possible stateless members are : Math.sin(double), * Math.PI. *

Examples of possible stateful members are : Math.random(), * System.currentTimeMillis(). * *

Stateless members of this library are always static members of the * classes, which define them. The inverse is generally not true. However, * this library goes as far as assuming that static members are stateless, * if this assumption does not hold for some of Your members it is possible to * mark them as stateful using the markStateDependent() method of * this class. * *

The most crucial difference between the two kind of members of this * library is that evaluation of stateless methods is attempted by JEL at * a compile time during the constants folding phase. */ public class Library { private HashMap> names; private HashMap dynIDs; private HashMap stateless; private HashMap>> dotClasses; private boolean noDotSecurity=false; public DVMap resolver; // directly accessed from EC.jj public HashMap cnmap; /** * Creates a library for JEL. *

See the three argument constructor for more info. * @param staticLib is the array of classes, whose public static * methods are exported. * @param dynamicLib is the array of classes, whose public virutal * methods are exported. * @deprecated Please us 5 argument constructor with unused arguments * set to null. This constructor is scheduled for removal in * JEL 1.0. */ public @Deprecated Library(Class[] staticLib,Class[] dynamicLib) { this(staticLib,dynamicLib,null); }; /** * Creates a library for JEL. *

See the three argument constructor for more info. * @param staticLib is the array of classes, whose public static * methods are exported. * @param dynamicLib is the array of classes, whose public virutal * methods are exported. * @param dotClasses Controls access to the "dot" operator on classes. * @deprecated Please us 5 argument constructor with unused arguments * set to null. This constructor is scheduled for removal in * JEL 1.0. */ public @Deprecated Library(Class[] staticLib,Class[] dynamicLib,Class[] dotClasses) { this(staticLib,dynamicLib,dotClasses,null); }; /** * Creates a library for JEL. *

See the three argument constructor for more info. * @param staticLib is the array of classes, whose public static * methods are exported. * @param dynamicLib is the array of classes, whose public virutal * methods are exported. * @param dotClasses Controls access to the "dot" operator on classes. * @param resolver Controls resolution of dynamic variables. * @deprecated Please us 5 argument constructor with unused arguments * set to null. This constructor is scheduled for removal in * JEL 1.0. */ public @Deprecated Library(Class[] staticLib,Class[] dynamicLib,Class[] dotClasses, DVMap resolver) { this(staticLib,dynamicLib,dotClasses,resolver,null); }; /** * Creates a library for JEL. *

The following should be kept in mind when constructing a library: *

    *
  1. This constructor may throw IllegalArgumentException if it does not * like something in Your library. The requirements to the method names * are somewhat more strict, than in Java because members of several * classes can be merged in root namespace. *
  2. When calling the * CompiledExpression.evaluate(Object[] dynalib) of the * expression, using dynamic library methods it is needed to pass as * dynalib parameter the array of objects, of the classes * _exactly_ in the same order as they are appearing in the * dynamicLib parameter of this constructor. If You do not * allow to call dynamic methods (there is no sense, then, to use a compiler) * it is possible to pass null istead of dynalib. *
  3. Generally speaking, this class should not let You to create wrong * libraries. It's methods will throw exceptions, return false's , * ignore Your actions,... ;) *
* If methods in the library classes conflict with each other, the last * conflicting method will be skipped. You will not get any messages unless * debugging is ON (see gnu.jel.debug.Debug.enabled). This is * done to avoid unnecessary error messages in the production code of the * compiler. *

The array (dotClasses), which is the third argument of * this constructor determines how (and whether) to compile the * dot operators encountered in expressions. These operators are * of the form <object>.(<method>|<field>), * which means to call method (access field) * of an <object>. There can be three types of the behaviour: *

1) dot operators are prohibited (dotClasses==null), * this is behaviour of older version of JEL. *

2) dot operators are allowed on all classes * (dotClasses==new Class[0], an empty array). * Depending on the types of objects returned by the static/dynamic library * classes this may pose a security risk. *

3) dot operators are allowed only on some classes. This is achieved * by listing these classes in the dotClasses array. * @param staticLib is the array of classes, whose public static * methods are exported. * @param dynamicLib is the array of classes, whose public virutal * methods are exported. * @param dotClasses is the array of classes on which the dot ('.') * operation is allowed. * @param resolver is the object used to resolve the names. * @param cnmap Maps class names into classes for non-primitive type casts. */ public Library(Class[] staticLib,Class[] dynamicLib,Class[] dotClasses, DVMap resolver,HashMap cnmap) { this.cnmap=cnmap; this.resolver=resolver; if (dotClasses==null) { this.dotClasses=null; } else { noDotSecurity=(dotClasses.length==0); this.dotClasses= new HashMap>>(); // hash the names Class[] temp=new Class[1]; for(int i=0;i>(); dynIDs = new HashMap(); stateless=new HashMap(); if (staticLib!=null) rehash(staticLib,names,null,stateless); if (dynamicLib!=null) rehash(dynamicLib,names,dynIDs,null); }; private void rehash(Class cls) { HashMap> tempNames=new HashMap>(); Class[] temp=new Class[1]; temp[0]=cls; // rehash(temp,tempNames,null,new HashMap()); rehash(temp,tempNames,new HashMap(),null); dotClasses.put(cls,tempNames); }; private static void rehash(Class[] arr, HashMap>hashedNames, HashMap dynIDs, HashMap stateless) { for (int i=0; i0) { // static if ((stateless!=null) && rehash(hashedNames,m)) stateless.put(m,Boolean.TRUE); } else { // not static if ((dynIDs!=null) && rehash(hashedNames,m)) dynIDs.put(m,dynID); }; }; }; }; private static boolean rehash(HashMap> hashedNames, Member m) { String name=m.getName(); String signature=getSignature(m); // for the purpose of matching fields behave like methods with no // arguments if (isField(m)) signature="()"+signature; HashMap signatures=hashedNames.get(name); if (signatures==null) { // No method with this name was added HashMap signatures_new=new HashMap(); signatures_new.put(signature,m); hashedNames.put(name,signatures_new); return true; }; // Name exists in the library, check for possible signature conflict. Object conflicting_method=signatures.get(signature); if (conflicting_method==null) { // No conflict signatures.put(signature,m); return true; }; // if (Debug.enabled) { // Debug.println("Conflict was detected during the library "+ // "initialization."+ // " Conflicting "+"\""+name+signature+ // "\", conflicting :"+ conflicting_method+" and "+m+" ."); // }; // If no debug then the method is ignored. return false; }; /** * This method marks a static member as having the internal state. *

If java.lang.Math is included into the library it is * necessary to mark java.lang.Math.random() as having the * state. This can be done by calling * markStateDependent("random",null). *

Please specify parameters as close as possible, otherwise you can * accidentally mark another function. * @param name is the function name. * @param params are the possible invocation parameters of the function. * @exception CompilationException if the method can't be resolved */ public void markStateDependent(String name, Class[] params) throws CompilationException { Object m=getMember(null,name,params); Object removed=stateless.remove(m); if (Debug.enabled) Debug.check(removed!=null,"State dependent methos \""+m+ "\"is made state dependend again."); }; /** * Used to check if the given method is stateless. * @param o is method or field to check. * @return true if the method is stateless and can be invoked at * compile time. */ public boolean isStateless(Member o) { return stateless.containsKey(o); }; /** * Searches the namespace defined by this library object for method or field. *

The method with the same name, and closest (convertible) parameter * types is returned. If there are several methods the most specific one * is used. *

Ambiguities are detected. Example of detectable ambiguity:
* you ask for a call someName(int, int), but there are two * applicable methods someName(int, double) and * someName(double, int). Requirements to parameter types * of both can be satisfied by _widening_ conversions. Thus, there * is no most specific method of these two in terms of Java Language * Specification (15.11.2.2). It means, that an ambiguity is present, * and null will be returned. *

Java compiler normally would not allow to define such ambiguous * methods in the same class. However, as this library is assembled from * several Java classes, such ambiguities can happen, and should be * detected. * @param container the class to search the method within, if null * the root namespace is searched. * @param name is the name of the method to find. * @param params are the types of formal parameters in the method invocation. * @return the method/field object of the resolved method/field. * @exception CompilationException if the method can't be resolved */ public Member getMember(Class container,String name,Class[] params) throws CompilationException { HashMap> hashedMembers=names; // dot operator security if (container!=null) { if (dotClasses==null) // dot operator is prohibited throw new CompilationException(11,null); else if (! (noDotSecurity || dotClasses.containsKey(container))) { // dot is not allowed in this particular class Object[] paramsExc={container}; throw new CompilationException(12,paramsExc); }; if ((hashedMembers=dotClasses.get(container))==null) { rehash(container); hashedMembers=dotClasses.get(container); }; }; HashMap signatures=hashedMembers.get(name); // System.out.print("RESOLVING: "); // System.out.println(describe(name,params)); if (signatures==null) { // name is not found Object[] paramsExc={name,container}; throw new CompilationException(container==null?5:6,paramsExc); }; // Choose applicable methods ArrayList applicable_methods=new ArrayList(); for(Member cm: signatures.values()) { Class[] cp=getParameterTypes(cm); boolean applicable=false; if (params!=null) { // if no parameters requested if (cp.length==params.length) { // If number of arguments matches applicable=true; for(int i=0;((i e=applicable_methods.iterator(); Member most_specific=e.next(); Class[] most_specific_params=getParameterTypes(most_specific); // System.out.println("--- APPLICABLE METHODS ---"); // System.out.println(most_specific.getName()+ // ClassFile.getSignature(most_specific)); while (e.hasNext()) { Member cm= e.next(); Class[] cp=getParameterTypes(cm); boolean moreSpecific=true; boolean lessSpecific=true; // System.out.println(cm.getName()+ClassFile.getSignature(cm)); for(int i=0; i ID's are used to locate the pointers to the objects, implementing * dynamic methods, in the array, argument of evaluate(Object[]) function. * @param m method to get an ID of. * @return the ID of the method or -1 if the method is static. * @exception NullPointerException if method is not a dynamic method of * this library. */ public int getDynamicMethodClassID(Member m) { Integer id=dynIDs.get(m); if (id==null) return -1; return id.intValue(); }; /** * Used to get return type of a class member. *

The type of a method is its return type, the type of a constructor is * void. * @param m member whose type is to be determined * @return type of the member */ public static Class getType(Member m) { if (m instanceof Method) return ((Method)m).getReturnType(); if (m instanceof Field) return ((Field)m).getType(); if (m instanceof LocalField) return ((LocalField)m).getType(); // otherwise it must be java.lang.reflect.Constructor if (Debug.enabled) Debug.check(m instanceof java.lang.reflect.Constructor); return OP.specialTypes[9]; // java.lang.reflect.Void.TYPE }; /** * Used to get types of formal parameters of a member. *

The reference to the class instance "this" is not counted by * this method. * @param m member whose formal parameters are to be obtained * @return array of formal parameter types (empty array if none). */ public static Class[] getParameterTypes(Member m) { if (m instanceof Method) return ((Method)m).getParameterTypes(); if (m instanceof LocalMethod) return ((LocalMethod)m).getParameterTypes(); if (m instanceof Constructor) return ((Constructor)m).getParameterTypes(); if (Debug.enabled) Debug.check((m instanceof Field)||(m instanceof LocalField)); return new Class[0]; }; /** * Computes signature of the given member. * @param m the member to compute the sugnature of. * @return the signature. */ public static String getSignature(Member m) { StringBuilder signature=new StringBuilder(); if (!isField(m)) { Class parameters[]=getParameterTypes(m); signature.append('('); for(int i=0;i The signature of the class (Field descriptor) is the string and * it's format is described in the paragraph 4.3.2 of the Java VM * specification (ISBN 0-201-63451-1). *

The same can be done using java.lang.Class.getName() by * converting it's result into the "historical form". *

This utility method can be used outside of the JEL package * it does not involve any JEL specific assumptions and should follow * JVM Specification precisely. * @param cls is the class to compute the signature of. Can be primitive or * array type. * @return the class signature. */ public static String getSignature(Class cls) { return appendSignature(new StringBuilder(),cls).toString(); }; private static StringBuilder appendSignature(StringBuilder buff, Class cls) { if (cls.isPrimitive()) { int tid; buff.append((tid=OP.typeID(cls))>9?'L':"ZBCSIJFDLV".charAt(tid)); } else if (cls.isArray()) { buff.append('['); appendSignature(buff,cls.getComponentType()); } else { // just a class buff.append('L'); appendHistoricalForm(buff,cls.getName()); buff.append(';'); }; return buff; }; public static String toHistoricalForm(String className) { return appendHistoricalForm(new StringBuilder(),className).toString(); }; private static StringBuilder appendHistoricalForm(StringBuilder buff, String className) { int namelen=className.length(); for(int i=0;i





© 2015 - 2025 Weber Informatics LLC | Privacy Policy