Open Source Repository

Home /beanshell/bsh-2.0b4 | Repository Home



bsh/ClassGeneratorImpl.java
package bsh;

import java.io.*;
import java.util.*;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

/**
  This class is an implementation of the ClassGenerator interface which
  contains generally bsh related code.  The actual bytecode generation is
  done by ClassGeneratorUtil.
  
  @author Pat Niemeyer ([email protected])
*/
public class ClassGeneratorImpl extends ClassGenerator
{
  public Class generateClass
    String name, Modifiers modifiers, 
    Class [] interfaces, Class superClass, BSHBlock block, 
    boolean isInterface, CallStack callstack, Interpreter interpreter 
  )
    throws EvalError
  {
    // Delegate to the static method
    return generateClassImplname, modifiers, interfaces, superClass,
      block, isInterface, callstack, interpreter );
  }

  public Object invokeSuperclassMethod(
    BshClassManager bcm, Object instance, String methodName, Object [] args
  )
        throws UtilEvalError, ReflectError, InvocationTargetException
  {
    // Delegate to the static method
    return invokeSuperclassMethodImplbcm, instance, methodName, args );
  }

  /**
    Change the parent of the class instance namespace.
    This is currently used for inner class support.
    Note: This method will likely be removed in the future.
  */
  // This could be static
  public void setInstanceNameSpaceParent
    Object instance, String className, NameSpace parent )
  {
    This ithis = 
      ClassGeneratorUtil.getClassInstanceThisinstance, className );
    ithis.getNameSpace().setParentparent );
  }

  /**
    If necessary, parse the BSHBlock for for the class definition and
    generate the class using ClassGeneratorUtil.
    This method also initializes the static block namespace and sets it
    in the class.
  */
  public static Class generateClassImpl
    String name, Modifiers modifiers, 
    Class [] interfaces, Class superClass, BSHBlock block, 
    boolean isInterface, CallStack callstack, Interpreter interpreter 
  )
    throws EvalError
  {
    // Scripting classes currently requires accessibility
    // This can be eliminated with a bit more work.
    try {
      Capabilities.setAccessibilitytrue );
    catch Capabilities.Unavailable e )
    {
      throw new EvalError(
        "Defining classes currently requires reflective Accessibility.",
        block, callstack );
    }

    NameSpace enclosingNameSpace = callstack.top();
    String packageName = enclosingNameSpace.getPackage();
    String className =  enclosingNameSpace.isClass ?
      enclosingNameSpace.getName()+"$"+name : name;
    String fqClassName =
      packageName == null ? className : packageName + "." + className;
    String bshStaticFieldName = ClassGeneratorUtil.BSHSTATIC+className;

    BshClassManager bcm = interpreter.getClassManager();
    // Race condition here...
    bcm.definingClassfqClassName );

    // Create the class static namespace
    NameSpace classStaticNameSpace =
      new NameSpaceenclosingNameSpace, className);
    classStaticNameSpace.isClass = true;

    callstack.pushclassStaticNameSpace );

    // Evaluate any inner class class definitions in the block
    // effectively recursively call this method for contained classes first
    block.evalBlock(
      callstack, interpreter, true/*override*/,
      ClassNodeFilter.CLASSCLASSES );

    // Generate the type for our class
    Variable [] variables =
      getDeclaredVariablesblock, callstack, interpreter, packageName );
    DelayedEvalBshMethod [] methods =
      getDeclaredMethodsblock, callstack, interpreter, packageName );

    // Create the class generator, which encapsulates all knowledge of the
    // structure of the class
    ClassGeneratorUtil classGenerator =  new ClassGeneratorUtil(
      modifiers, className, packageName, superClass, interfaces,
      variables, methods, isInterface );

    // Check for existing class (saved class file)
    Class clas = bcm.getAssociatedClassfqClassName );

    // If the class isn't there then generate it.
    // Else just let it be initialized below.
    if clas == null )
    {
      // generate bytecode, optionally with static init hooks to
      // bootstrap the interpreter
      byte [] code = classGenerator.generateClass(
        Interpreter.getSaveClasses()/*init code*/ );

      if Interpreter.getSaveClasses() )
        saveClassesclassName, code );
      else
        clas = bcm.defineClassfqClassName, code );
    }

    // If we're just saving clases then don't actually execute the static
    // code for the class here.
    if !Interpreter.getSaveClasses() )
    {
      // Let the class generator install hooks relating to the structure of
      // the class into the class static namespace.  e.g. the constructor
      // array.  This is necessary whether we are generating code or just
      // reinitializing a previously generated class.
      classGenerator.initStaticNameSpace(
        classStaticNameSpace, block/*instance initializer*/ );

      // import the unqualified class name into parent namespace
      enclosingNameSpace.importClassfqClassName.replace('$','.') );

      // Give the static space its class static import
      // important to do this after all classes are defined
      classStaticNameSpace.setClassStaticclas );

      // evaluate the static portion of the block in the static space
      block.evalBlock(
        callstack, interpreter, true/*override*/,
        ClassNodeFilter.CLASSSTATIC );

      if !clas.isInterface() )
        installStaticBlockclas, bshStaticFieldName,
          classStaticNameSpace, interpreter );
    }

    callstack.pop();

    bcm.doneDefiningClassfqClassName );
    return clas;
  }

  private static void installStaticBlock(
    Class genClass, String bshStaticFieldName,
    NameSpace classStaticNameSpace, Interpreter interpreter )
  {
    // Set the static bsh This callback
    try {
      LHS lhs = Reflect.getLHSStaticFieldgenClass, bshStaticFieldName );
      lhs.assign(
        classStaticNameSpace.getThisinterpreter )false/*strict*/ );
    catch Exception e ) {
      throw new InterpreterError("Error in class gen setup: "+e );
    }
  }

  private static void saveClassesString className, byte[] code )
  {
    String dir = Interpreter.getSaveClassesDir();
    if dir != null )
    try {
      FileOutputStream out=
        new FileOutputStreamdir+"/"+className+".class" );
      out.write(code);
      out.close();
    catch IOException e ) {
      e.printStackTrace();
    }
  }

  static Variable [] getDeclaredVariables
    BSHBlock body, CallStack callstack, Interpreter interpreter, 
    String defaultPackage 
  
  {
    List vars = new ArrayList();
    forint child=0; child<body.jjtGetNumChildren(); child++ )
    {
      SimpleNode node = (SimpleNode)body.jjtGetChild(child);
      if node instanceof BSHTypedVariableDeclaration )
      {
        BSHTypedVariableDeclaration tvd = 
          (BSHTypedVariableDeclaration)node;
        Modifiers modifiers = tvd.modifiers;

        String type = tvd.getTypeDescriptor
          callstack, interpreter, defaultPackage );

        BSHVariableDeclarator [] vardec = tvd.getDeclarators();
        forint i = 0; i< vardec.length; i++)
        {
          String name = vardec[i].name;
          try {
            Variable var = new Variable
              name, type, null/*value*/, modifiers );
            vars.addvar );
          catch UtilEvalError e ) {
            // value error shouldn't happen
          }
        }
      }
    }

    return (Variable [])vars.toArraynew Variable[0] );
  }

  static DelayedEvalBshMethod [] getDeclaredMethods
    BSHBlock body, CallStack callstack, Interpreter interpreter,
    String defaultPackage 
  )
  {
    List methods = new ArrayList();
    forint child=0; child<body.jjtGetNumChildren(); child++ )
    {
      SimpleNode node = (SimpleNode)body.jjtGetChild(child);
      if node instanceof BSHMethodDeclaration )
      {
        BSHMethodDeclaration md = (BSHMethodDeclaration)node;
        md.insureNodesParsed();
        Modifiers modifiers = md.modifiers;
        String name = md.name;
        String returnType = md.getReturnTypeDescriptor
          callstack, interpreter, defaultPackage );
        BSHReturnType returnTypeNode = md.getReturnTypeNode();
        BSHFormalParameters paramTypesNode = md.paramsNode;
        String [] paramTypes = paramTypesNode.getTypeDescriptors
          callstack, interpreter, defaultPackage );

        DelayedEvalBshMethod bm = new DelayedEvalBshMethod
          name, 
          returnType, returnTypeNode,
          md.paramsNode.getParamNames()
          paramTypes, paramTypesNode,
          md.blockNode, null/*declaringNameSpace*/,
          modifiers, callstack, interpreter 
        );

        methods.addbm );
      }
    }

    return (DelayedEvalBshMethod [])methods.toArray
      new DelayedEvalBshMethod[0] );
  }

  /**
    A node filter that filters nodes for either a class body static 
    initializer or instance initializer.  In the static case only static 
    members are passed, etc.  
  */
  static class ClassNodeFilter implements BSHBlock.NodeFilter
  {
    public static final int STATIC=0, INSTANCE=1, CLASSES=2;

    public static ClassNodeFilter CLASSSTATIC = 
      new ClassNodeFilterSTATIC );
    public static ClassNodeFilter CLASSINSTANCE = 
      new ClassNodeFilterINSTANCE );
    public static ClassNodeFilter CLASSCLASSES = 
      new ClassNodeFilterCLASSES );

    int context;

    private ClassNodeFilterint context ) { this.context = context; }

    public boolean isVisibleSimpleNode node 
    {
      if context == CLASSES )
        return node instanceof BSHClassDeclaration;

      // Only show class decs in CLASSES
      if node instanceof BSHClassDeclaration )
        return false;

      if context == STATIC )
        return isStaticnode );

      if context == INSTANCE )
        return !isStaticnode );

      // ALL
      return true;
    }

    boolean isStaticSimpleNode node 
    {
      if node instanceof BSHTypedVariableDeclaration )
        return ((BSHTypedVariableDeclaration)node).modifiers != null
          && ((BSHTypedVariableDeclaration)node).modifiers
            .hasModifier("static");

      if node instanceof BSHMethodDeclaration )
        return ((BSHMethodDeclaration)node).modifiers != null
          && ((BSHMethodDeclaration)node).modifiers
            .hasModifier("static");

      // need to add static block here
      if node instanceof BSHBlock)
        return ((BSHBlock)node).isStatic;

      return false;
    }
  }

  public static Object invokeSuperclassMethodImpl(
    BshClassManager bcm, Object instance, String methodName, Object [] args
  )
        throws UtilEvalError, ReflectError, InvocationTargetException
  {
    String superName = ClassGeneratorUtil.BSHSUPER+methodName;
    
    // look for the specially named super delegate method
    Class clas = instance.getClass();
    Method superMethod = Reflect.resolveJavaMethod(
      bcm, clas, superName, Types.getTypes(args)false/*onlyStatic*/ );
    if superMethod != null )
      return Reflect.invokeMethod(
        superMethod, instance, args );

    // No super method, try to invoke regular method
    // could be a superfluous "super." which is legal.
    Class superClass = clas.getSuperclass();
    superMethod = Reflect.resolveExpectedJavaMethod(
      bcm, superClass, instance, methodName, args, 
      false/*onlyStatic*/ );
    return Reflect.invokeMethodsuperMethod, instance, args );
  }

}