Package clojure.lang

Source Code of clojure.lang.Compiler$MethodParamExpr

/**
*   Copyright (c) Rich Hickey. All rights reserved.
*   The use and distribution terms for this software are covered by the
*   Eclipse Public License 1.0 (http://opensource.org/licenses/eclipse-1.0.php)
*   which can be found in the file epl-v10.html at the root of this distribution.
*   By using this software in any fashion, you are agreeing to be bound by
*    the terms of this license.
*   You must not remove this notice, or any other, from this software.
**/

/* rich Aug 21, 2007 */

package clojure.lang;

//*

import clojure.asm.*;
import clojure.asm.commons.Method;
import clojure.asm.commons.GeneratorAdapter;
//*/
/*

import org.objectweb.asm.*;
import org.objectweb.asm.commons.Method;
import org.objectweb.asm.commons.GeneratorAdapter;
import org.objectweb.asm.util.TraceClassVisitor;
import org.objectweb.asm.util.CheckClassAdapter;
//*/

import java.io.*;
import java.util.*;
import java.lang.reflect.Constructor;
import java.lang.reflect.Modifier;

public class Compiler implements Opcodes{

static final Symbol DEF = Symbol.create("def");
static final Symbol LOOP = Symbol.create("loop*");
static final Symbol RECUR = Symbol.create("recur");
static final Symbol IF = Symbol.create("if");
static final Symbol LET = Symbol.create("let*");
static final Symbol LETFN = Symbol.create("letfn*");
static final Symbol DO = Symbol.create("do");
static final Symbol FN = Symbol.create("fn*");
static final Symbol QUOTE = Symbol.create("quote");
static final Symbol THE_VAR = Symbol.create("var");
static final Symbol DOT = Symbol.create(".");
static final Symbol ASSIGN = Symbol.create("set!");
//static final Symbol TRY_FINALLY = Symbol.create("try-finally");
static final Symbol TRY = Symbol.create("try");
static final Symbol CATCH = Symbol.create("catch");
static final Symbol FINALLY = Symbol.create("finally");
static final Symbol THROW = Symbol.create("throw");
static final Symbol MONITOR_ENTER = Symbol.create("monitor-enter");
static final Symbol MONITOR_EXIT = Symbol.create("monitor-exit");
static final Symbol IMPORT = Symbol.create("clojure.core", "import*");
//static final Symbol INSTANCE = Symbol.create("instance?");
static final Symbol DEFTYPE = Symbol.create("deftype*");
static final Symbol CASE = Symbol.create("case*");

//static final Symbol THISFN = Symbol.create("thisfn");
static final Symbol CLASS = Symbol.create("Class");
static final Symbol NEW = Symbol.create("new");
static final Symbol THIS = Symbol.create("this");
static final Symbol REIFY = Symbol.create("reify*");
//static final Symbol UNQUOTE = Symbol.create("unquote");
//static final Symbol UNQUOTE_SPLICING = Symbol.create("unquote-splicing");
//static final Symbol SYNTAX_QUOTE = Symbol.create("clojure.core", "syntax-quote");
static final Symbol LIST = Symbol.create("clojure.core", "list");
static final Symbol HASHMAP = Symbol.create("clojure.core", "hash-map");
static final Symbol VECTOR = Symbol.create("clojure.core", "vector");
static final Symbol IDENTITY = Symbol.create("clojure.core", "identity");

static final Symbol _AMP_ = Symbol.create("&");
static final Symbol ISEQ = Symbol.create("clojure.lang.ISeq");

static final Keyword inlineKey = Keyword.intern(null, "inline");
static final Keyword inlineAritiesKey = Keyword.intern(null, "inline-arities");

static final Keyword volatileKey = Keyword.intern(null, "volatile");
static final Keyword implementsKey = Keyword.intern(null, "implements");
static final String COMPILE_STUB_PREFIX = "compile__stub";

static final Keyword protocolKey = Keyword.intern(null, "protocol");
static final Keyword onKey = Keyword.intern(null, "on");

static final Symbol NS = Symbol.create("ns");
static final Symbol IN_NS = Symbol.create("in-ns");

//static final Symbol IMPORT = Symbol.create("import");
//static final Symbol USE = Symbol.create("use");

//static final Symbol IFN = Symbol.create("clojure.lang", "IFn");

static final public IPersistentMap specials = PersistentHashMap.create(
    DEF, new DefExpr.Parser(),
    LOOP, new LetExpr.Parser(),
    RECUR, new RecurExpr.Parser(),
    IF, new IfExpr.Parser(),
    CASE, new CaseExpr.Parser(),
    LET, new LetExpr.Parser(),
    LETFN, new LetFnExpr.Parser(),
    DO, new BodyExpr.Parser(),
    FN, null,
    QUOTE, new ConstantExpr.Parser(),
    THE_VAR, new TheVarExpr.Parser(),
    IMPORT, new ImportExpr.Parser(),
    DOT, new HostExpr.Parser(),
    ASSIGN, new AssignExpr.Parser(),
    DEFTYPE, new NewInstanceExpr.DeftypeParser(),
    REIFY, new NewInstanceExpr.ReifyParser(),
//    TRY_FINALLY, new TryFinallyExpr.Parser(),
TRY, new TryExpr.Parser(),
THROW, new ThrowExpr.Parser(),
MONITOR_ENTER, new MonitorEnterExpr.Parser(),
MONITOR_EXIT, new MonitorExitExpr.Parser(),
//    INSTANCE, new InstanceExpr.Parser(),
//    IDENTICAL, new IdenticalExpr.Parser(),
//THISFN, null,
CATCH, null,
FINALLY, null,
//    CLASS, new ClassExpr.Parser(),
NEW, new NewExpr.Parser(),
//    UNQUOTE, null,
//    UNQUOTE_SPLICING, null,
//    SYNTAX_QUOTE, null,
_AMP_, null
);

private static final int MAX_POSITIONAL_ARITY = 20;
private static final Type OBJECT_TYPE;
private static final Type KEYWORD_TYPE = Type.getType(Keyword.class);
private static final Type VAR_TYPE = Type.getType(Var.class);
private static final Type SYMBOL_TYPE = Type.getType(Symbol.class);
//private static final Type NUM_TYPE = Type.getType(Num.class);
private static final Type IFN_TYPE = Type.getType(IFn.class);
private static final Type AFUNCTION_TYPE = Type.getType(AFunction.class);
private static final Type RT_TYPE = Type.getType(RT.class);
final static Type CLASS_TYPE = Type.getType(Class.class);
final static Type NS_TYPE = Type.getType(Namespace.class);
final static Type UTIL_TYPE = Type.getType(Util.class);
final static Type REFLECTOR_TYPE = Type.getType(Reflector.class);
final static Type THROWABLE_TYPE = Type.getType(Throwable.class);
final static Type BOOLEAN_OBJECT_TYPE = Type.getType(Boolean.class);
final static Type IPERSISTENTMAP_TYPE = Type.getType(IPersistentMap.class);
final static Type IOBJ_TYPE = Type.getType(IObj.class);

private static final Type[][] ARG_TYPES;
private static final Type[] EXCEPTION_TYPES = {Type.getType(Exception.class)};

static
  {
  OBJECT_TYPE = Type.getType(Object.class);
  ARG_TYPES = new Type[MAX_POSITIONAL_ARITY + 2][];
  for(int i = 0; i <= MAX_POSITIONAL_ARITY; ++i)
    {
    Type[] a = new Type[i];
    for(int j = 0; j < i; j++)
      a[j] = OBJECT_TYPE;
    ARG_TYPES[i] = a;
    }
  Type[] a = new Type[MAX_POSITIONAL_ARITY + 1];
  for(int j = 0; j < MAX_POSITIONAL_ARITY; j++)
    a[j] = OBJECT_TYPE;
  a[MAX_POSITIONAL_ARITY] = Type.getType("[Ljava/lang/Object;");
  ARG_TYPES[MAX_POSITIONAL_ARITY + 1] = a;


  }


//symbol->localbinding
static final public Var LOCAL_ENV = Var.create(null);

//vector<localbinding>
static final public Var LOOP_LOCALS = Var.create();

//Label
static final public Var LOOP_LABEL = Var.create();

//vector<object>
static final public Var CONSTANTS = Var.create();

//IdentityHashMap
static final public Var CONSTANT_IDS = Var.create();

//vector<keyword>
static final public Var KEYWORD_CALLSITES = Var.create();

//vector<var>
static final public Var PROTOCOL_CALLSITES = Var.create();

//vector<var>
static final public Var VAR_CALLSITES = Var.create();

//keyword->constid
static final public Var KEYWORDS = Var.create();

//var->constid
static final public Var VARS = Var.create();

//FnFrame
static final public Var METHOD = Var.create(null);

//null or not
static final public Var IN_CATCH_FINALLY = Var.create(null);

//DynamicClassLoader
static final public Var LOADER = Var.create();

//String
static final public Var SOURCE = Var.intern(Namespace.findOrCreate(Symbol.create("clojure.core")),
                                            Symbol.create("*source-path*"), "NO_SOURCE_FILE");

//String
static final public Var SOURCE_PATH = Var.intern(Namespace.findOrCreate(Symbol.create("clojure.core")),
                                                 Symbol.create("*file*"), "NO_SOURCE_PATH");

//String
static final public Var COMPILE_PATH = Var.intern(Namespace.findOrCreate(Symbol.create("clojure.core")),
                                                  Symbol.create("*compile-path*"), null);
//boolean
static final public Var COMPILE_FILES = Var.intern(Namespace.findOrCreate(Symbol.create("clojure.core")),
                                                   Symbol.create("*compile-files*"), Boolean.FALSE);

static final public Var INSTANCE = Var.intern(Namespace.findOrCreate(Symbol.create("clojure.core")),
                                            Symbol.create("instance?"));

static final public Var ADD_ANNOTATIONS = Var.intern(Namespace.findOrCreate(Symbol.create("clojure.core")),
                                            Symbol.create("add-annotations"));

//Integer
static final public Var LINE = Var.create(0);

//Integer
static final public Var LINE_BEFORE = Var.create(0);
static final public Var LINE_AFTER = Var.create(0);

//Integer
static final public Var NEXT_LOCAL_NUM = Var.create(0);

//Integer
static final public Var RET_LOCAL_NUM = Var.create();


static final public Var COMPILE_STUB_SYM = Var.create(null);
static final public Var COMPILE_STUB_CLASS = Var.create(null);


//PathNode chain
static final public Var CLEAR_PATH = Var.create(null);

//tail of PathNode chain
static final public Var CLEAR_ROOT = Var.create(null);

//LocalBinding -> Set<LocalBindingExpr>
static final public Var CLEAR_SITES = Var.create(null);

    public enum C{
  STATEMENT,  //value ignored
  EXPRESSION, //value required
  RETURN,      //tail position relative to enclosing recur frame
  EVAL
}

interface Expr{
  Object eval() throws Exception;

  void emit(C context, ObjExpr objx, GeneratorAdapter gen);

  boolean hasJavaClass() throws Exception;

  Class getJavaClass() throws Exception;
}

public static abstract class UntypedExpr implements Expr{

  public Class getJavaClass(){
    throw new IllegalArgumentException("Has no Java class");
  }

  public boolean hasJavaClass(){
    return false;
  }
}

interface IParser{
  Expr parse(C context, Object form) throws Exception;
}

static boolean isSpecial(Object sym){
  return specials.containsKey(sym);
}

static Symbol resolveSymbol(Symbol sym){
  //already qualified or classname?
  if(sym.name.indexOf('.') > 0)
    return sym;
  if(sym.ns != null)
    {
    Namespace ns = namespaceFor(sym);
    if(ns == null || ns.name.name == sym.ns)
      return sym;
    return Symbol.create(ns.name.name, sym.name);
    }
  Object o = currentNS().getMapping(sym);
  if(o == null)
    return Symbol.intern(currentNS().name.name, sym.name);
  else if(o instanceof Class)
    return Symbol.intern(null, ((Class) o).getName());
  else if(o instanceof Var)
      {
      Var v = (Var) o;
      return Symbol.create(v.ns.name.name, v.sym.name);
      }
  return null;

}

static class DefExpr implements Expr{
  public final Var var;
  public final Expr init;
  public final Expr meta;
  public final boolean initProvided;
  public final String source;
  public final int line;
  final static Method bindRootMethod = Method.getMethod("void bindRoot(Object)");
  final static Method setTagMethod = Method.getMethod("void setTag(clojure.lang.Symbol)");
  final static Method setMetaMethod = Method.getMethod("void setMeta(clojure.lang.IPersistentMap)");
  final static Method symcreate = Method.getMethod("clojure.lang.Symbol create(String, String)");

  public DefExpr(String source, int line, Var var, Expr init, Expr meta, boolean initProvided){
    this.source = source;
    this.line = line;
    this.var = var;
    this.init = init;
    this.meta = meta;
    this.initProvided = initProvided;
  }

    private boolean includesExplicitMetadata(MapExpr expr) {
        for(int i=0; i < expr.keyvals.count(); i += 2)
            {
                Keyword k  = ((KeywordExpr) expr.keyvals.nth(i)).k;
                if ((k != RT.FILE_KEY) &&
                    (k != RT.DECLARED_KEY) &&
                    (k != RT.LINE_KEY))
                    return true;
            }
        return false;
    }

    public Object eval() throws Exception{
    try
      {
      if(initProvided)
        {
//      if(init instanceof FnExpr && ((FnExpr) init).closes.count()==0)
//        var.bindRoot(new FnLoaderThunk((FnExpr) init,var));
//      else
        var.bindRoot(init.eval());
        }
      if(meta != null)
        {
                IPersistentMap metaMap = (IPersistentMap) meta.eval();
                if (initProvided || includesExplicitMetadata((MapExpr) meta))
            var.setMeta((IPersistentMap) meta.eval());
        }
      return var;
      }
    catch(Throwable e)
      {
      if(!(e instanceof CompilerException))
        throw new CompilerException(source, line, e);
      else
        throw (CompilerException) e;
      }
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    objx.emitVar(gen, var);
    if(meta != null)
      {
            if (initProvided || includesExplicitMetadata((MapExpr) meta))
                {
                gen.dup();
                meta.emit(C.EXPRESSION, objx, gen);
                gen.checkCast(IPERSISTENTMAP_TYPE);
                gen.invokeVirtual(VAR_TYPE, setMetaMethod);
                }
      }
    if(initProvided)
      {
      gen.dup();
      init.emit(C.EXPRESSION, objx, gen);
      gen.invokeVirtual(VAR_TYPE, bindRootMethod);
      }

    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass(){
    return true;
  }

  public Class getJavaClass(){
    return Var.class;
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object form) throws Exception{
      //(def x) or (def x initexpr)
      if(RT.count(form) > 3)
        throw new Exception("Too many arguments to def");
      else if(RT.count(form) < 2)
        throw new Exception("Too few arguments to def");
      else if(!(RT.second(form) instanceof Symbol))
          throw new Exception("First argument to def must be a Symbol");
      Symbol sym = (Symbol) RT.second(form);
      Var v = lookupVar(sym, true);
      if(v == null)
        throw new Exception("Can't refer to qualified var that doesn't exist");
      if(!v.ns.equals(currentNS()))
        {
        if(sym.ns == null)
          v = currentNS().intern(sym);
//          throw new Exception("Name conflict, can't def " + sym + " because namespace: " + currentNS().name +
//                              " refers to:" + v);
        else
          throw new Exception("Can't create defs outside of current ns");
        }
      IPersistentMap mm = sym.meta();
            Object source_path = SOURCE_PATH.get();
            source_path = source_path == null ? "NO_SOURCE_FILE" : source_path;
            mm = (IPersistentMap) RT.assoc(mm, RT.LINE_KEY, LINE.get()).assoc(RT.FILE_KEY, source_path);
      Expr meta = analyze(context == C.EVAL ? context : C.EXPRESSION, mm);
      return new DefExpr((String) SOURCE.deref(), (Integer) LINE.deref(),
                         v, analyze(context == C.EVAL ? context : C.EXPRESSION, RT.third(form), v.sym.name),
                         meta, RT.count(form) == 3);
    }
  }
}

public static class AssignExpr implements Expr{
  public final AssignableExpr target;
  public final Expr val;

  public AssignExpr(AssignableExpr target, Expr val){
    this.target = target;
    this.val = val;
  }

  public Object eval() throws Exception{
    return target.evalAssign(val);
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    target.emitAssign(context, objx, gen, val);
  }

  public boolean hasJavaClass() throws Exception{
    return val.hasJavaClass();
  }

  public Class getJavaClass() throws Exception{
    return val.getJavaClass();
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object frm) throws Exception{
      ISeq form = (ISeq) frm;
      if(RT.length(form) != 3)
        throw new IllegalArgumentException("Malformed assignment, expecting (set! target val)");
      Expr target = analyze(C.EXPRESSION, RT.second(form));
      if(!(target instanceof AssignableExpr))
        throw new IllegalArgumentException("Invalid assignment target");
      return new AssignExpr((AssignableExpr) target, analyze(C.EXPRESSION, RT.third(form)));
    }
  }
}

public static class VarExpr implements Expr, AssignableExpr{
  public final Var var;
  public final Object tag;
  final static Method getMethod = Method.getMethod("Object get()");
  final static Method setMethod = Method.getMethod("Object set(Object)");

  public VarExpr(Var var, Symbol tag){
    this.var = var;
    this.tag = tag != null ? tag : var.getTag();
  }

  public Object eval() throws Exception{
    return var.deref();
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    objx.emitVar(gen, var);
    gen.invokeVirtual(VAR_TYPE, getMethod);
    if(context == C.STATEMENT)
      {
      gen.pop();
      }
  }

  public boolean hasJavaClass(){
    return tag != null;
  }

  public Class getJavaClass() throws Exception{
    return HostExpr.tagToClass(tag);
  }

  public Object evalAssign(Expr val) throws Exception{
    return var.set(val.eval());
  }

  public void emitAssign(C context, ObjExpr objx, GeneratorAdapter gen,
                         Expr val){
    objx.emitVar(gen, var);
    val.emit(C.EXPRESSION, objx, gen);
    gen.invokeVirtual(VAR_TYPE, setMethod);
    if(context == C.STATEMENT)
      gen.pop();
  }
}

public static class TheVarExpr implements Expr{
  public final Var var;

  public TheVarExpr(Var var){
    this.var = var;
  }

  public Object eval() throws Exception{
    return var;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    objx.emitVar(gen, var);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass(){
    return true;
  }

  public Class getJavaClass() throws ClassNotFoundException{
    return Var.class;
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object form) throws Exception{
      Symbol sym = (Symbol) RT.second(form);
      Var v = lookupVar(sym, false);
      if(v != null)
        return new TheVarExpr(v);
      throw new Exception("Unable to resolve var: " + sym + " in this context");
    }
  }
}

public static class KeywordExpr implements Expr{
  public final Keyword k;

  public KeywordExpr(Keyword k){
    this.k = k;
  }

  public Object eval() throws Exception{
    return k;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    objx.emitKeyword(gen, k);
    if(context == C.STATEMENT)
      gen.pop();

  }

  public boolean hasJavaClass(){
    return true;
  }

  public Class getJavaClass() throws ClassNotFoundException{
    return Keyword.class;
  }
}

public static class ImportExpr implements Expr{
  public final String c;
  final static Method forNameMethod = Method.getMethod("Class forName(String)");
  final static Method importClassMethod = Method.getMethod("Class importClass(Class)");
  final static Method derefMethod = Method.getMethod("Object deref()");

  public ImportExpr(String c){
    this.c = c;
  }

  public Object eval() throws Exception{
    Namespace ns = (Namespace) RT.CURRENT_NS.deref();
    ns.importClass(RT.classForName(c));
    return null;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.getStatic(RT_TYPE,"CURRENT_NS",VAR_TYPE);
    gen.invokeVirtual(VAR_TYPE, derefMethod);
    gen.checkCast(NS_TYPE);
    gen.push(c);
    gen.invokeStatic(CLASS_TYPE, forNameMethod);
    gen.invokeVirtual(NS_TYPE, importClassMethod);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass(){
    return false;
  }

  public Class getJavaClass() throws ClassNotFoundException{
    throw new IllegalArgumentException("ImportExpr has no Java class");
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object form) throws Exception{
      return new ImportExpr((String) RT.second(form));
    }
  }
}

public static abstract class LiteralExpr implements Expr{
  abstract Object val();

  public Object eval(){
    return val();
  }
}

static interface AssignableExpr{
  Object evalAssign(Expr val) throws Exception;

  void emitAssign(C context, ObjExpr objx, GeneratorAdapter gen, Expr val);
}

static public interface MaybePrimitiveExpr extends Expr{
  public boolean canEmitPrimitive();
  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen);
}

static public abstract class HostExpr implements Expr, MaybePrimitiveExpr{
  final static Type BOOLEAN_TYPE = Type.getType(Boolean.class);
  final static Type CHAR_TYPE = Type.getType(Character.class);
  final static Type INTEGER_TYPE = Type.getType(Integer.class);
  final static Type LONG_TYPE = Type.getType(Long.class);
  final static Type FLOAT_TYPE = Type.getType(Float.class);
  final static Type DOUBLE_TYPE = Type.getType(Double.class);
  final static Type SHORT_TYPE = Type.getType(Short.class);
  final static Type BYTE_TYPE = Type.getType(Byte.class);
  final static Type NUMBER_TYPE = Type.getType(Number.class);

  final static Method charValueMethod = Method.getMethod("char charValue()");
  final static Method booleanValueMethod = Method.getMethod("boolean booleanValue()");

  final static Method charValueOfMethod = Method.getMethod("Character valueOf(char)");
  final static Method intValueOfMethod = Method.getMethod("Integer valueOf(int)");
  final static Method longValueOfMethod = Method.getMethod("Long valueOf(long)");
  final static Method floatValueOfMethod = Method.getMethod("Float valueOf(float)");
  final static Method doubleValueOfMethod = Method.getMethod("Double valueOf(double)");
  final static Method shortValueOfMethod = Method.getMethod("Short valueOf(short)");
  final static Method byteValueOfMethod = Method.getMethod("Byte valueOf(byte)");

  final static Method intValueMethod = Method.getMethod("int intValue()");
  final static Method longValueMethod = Method.getMethod("long longValue()");
  final static Method floatValueMethod = Method.getMethod("float floatValue()");
  final static Method doubleValueMethod = Method.getMethod("double doubleValue()");
  final static Method byteValueMethod = Method.getMethod("byte byteValue()");
  final static Method shortValueMethod = Method.getMethod("short shortValue()");

  final static Method fromIntMethod = Method.getMethod("clojure.lang.Num from(int)");
  final static Method fromLongMethod = Method.getMethod("clojure.lang.Num from(long)");
  final static Method fromDoubleMethod = Method.getMethod("clojure.lang.Num from(double)");


  //*
  public static void emitBoxReturn(ObjExpr objx, GeneratorAdapter gen, Class returnType){
    if(returnType.isPrimitive())
      {
      if(returnType == boolean.class)
        {
        Label falseLabel = gen.newLabel();
        Label endLabel = gen.newLabel();
        gen.ifZCmp(GeneratorAdapter.EQ, falseLabel);
        gen.getStatic(BOOLEAN_OBJECT_TYPE, "TRUE", BOOLEAN_OBJECT_TYPE);
        gen.goTo(endLabel);
        gen.mark(falseLabel);
        gen.getStatic(BOOLEAN_OBJECT_TYPE, "FALSE", BOOLEAN_OBJECT_TYPE);
//        NIL_EXPR.emit(C.EXPRESSION, fn, gen);
        gen.mark(endLabel);
        }
      else if(returnType == void.class)
        {
        NIL_EXPR.emit(C.EXPRESSION, objx, gen);
        }
      else if(returnType == char.class)
          {
          gen.invokeStatic(CHAR_TYPE, charValueOfMethod);
          }
        else
          {
          if(returnType == int.class)
          //gen.invokeStatic(NUM_TYPE, fromIntMethod);
            gen.invokeStatic(INTEGER_TYPE, intValueOfMethod);
          else if(returnType == float.class)
            {
            //gen.visitInsn(F2D);
            gen.invokeStatic(FLOAT_TYPE, floatValueOfMethod);
            //m = floatValueOfMethod;
            }
          else if(returnType == double.class)
              gen.invokeStatic(DOUBLE_TYPE, doubleValueOfMethod);
            else if(returnType == long.class)
                gen.invokeStatic(LONG_TYPE, longValueOfMethod);
              else if(returnType == byte.class)
                  gen.invokeStatic(BYTE_TYPE, byteValueOfMethod);
                else if(returnType == short.class)
                    gen.invokeStatic(SHORT_TYPE, shortValueOfMethod);
          }
      }
  }

  //*/
  public static void emitUnboxArg(ObjExpr objx, GeneratorAdapter gen, Class paramType){
    if(paramType.isPrimitive())
      {
      if(paramType == boolean.class)
        {
        gen.checkCast(BOOLEAN_TYPE);
        gen.invokeVirtual(BOOLEAN_TYPE, booleanValueMethod);
//        Label falseLabel = gen.newLabel();
//        Label endLabel = gen.newLabel();
//        gen.ifNull(falseLabel);
//        gen.push(1);
//        gen.goTo(endLabel);
//        gen.mark(falseLabel);
//        gen.push(0);
//        gen.mark(endLabel);
        }
      else if(paramType == char.class)
        {
        gen.checkCast(CHAR_TYPE);
        gen.invokeVirtual(CHAR_TYPE, charValueMethod);
        }
      else
        {
        Method m = intValueMethod;
        gen.checkCast(NUMBER_TYPE);
        if(paramType == int.class)
          m = intValueMethod;
        else if(paramType == float.class)
          m = floatValueMethod;
        else if(paramType == double.class)
            m = doubleValueMethod;
          else if(paramType == long.class)
              m = longValueMethod;
            else if(paramType == byte.class)
                m = byteValueMethod;
              else if(paramType == short.class)
                  m = shortValueMethod;
        gen.invokeVirtual(NUMBER_TYPE, m);
        }
      }
    else
      {
      gen.checkCast(Type.getType(paramType));
      }
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object frm) throws Exception{
      ISeq form = (ISeq) frm;
      //(. x fieldname-sym) or
      //(. x 0-ary-method)
      // (. x methodname-sym args+)
      // (. x (methodname-sym args?))
      if(RT.length(form) < 3)
        throw new IllegalArgumentException("Malformed member expression, expecting (. target member ...)");
      //determine static or instance
      //static target must be symbol, either fully.qualified.Classname or Classname that has been imported
      int line = (Integer) LINE.deref();
      String source = (String) SOURCE.deref();
      Class c = maybeClass(RT.second(form), false);
      //at this point c will be non-null if static
      Expr instance = null;
      if(c == null)
        instance = analyze(context == C.EVAL ? context : C.EXPRESSION, RT.second(form));
      boolean maybeField = RT.length(form) == 3 &&
                           (RT.third(form) instanceof Symbol
                  || RT.third(form) instanceof Keyword);
      if(maybeField && !(RT.third(form) instanceof Keyword))
        {
        Symbol sym = (Symbol) RT.third(form);
        if(c != null)
          maybeField = Reflector.getMethods(c, 0, munge(sym.name), true).size() == 0;
        else if(instance != null && instance.hasJavaClass() && instance.getJavaClass() != null)
          maybeField = Reflector.getMethods(instance.getJavaClass(), 0, munge(sym.name), false).size() == 0;
        }
      if(maybeField)    //field
        {
        Symbol sym = (RT.third(form) instanceof Keyword)?
                     ((Keyword)RT.third(form)).sym
              :(Symbol) RT.third(form);
        Symbol tag = tagOf(form);
        if(c != null) {
          return new StaticFieldExpr(line, c, munge(sym.name), tag);
        } else
          return new InstanceFieldExpr(line, instance, munge(sym.name), tag);
        }
      else
        {
        ISeq call = (ISeq) ((RT.third(form) instanceof ISeq) ? RT.third(form) : RT.next(RT.next(form)));
        if(!(RT.first(call) instanceof Symbol))
          throw new IllegalArgumentException("Malformed member expression");
        Symbol sym = (Symbol) RT.first(call);
        Symbol tag = tagOf(form);
        PersistentVector args = PersistentVector.EMPTY;
        for(ISeq s = RT.next(call); s != null; s = s.next())
          args = args.cons(analyze(context == C.EVAL ? context : C.EXPRESSION, s.first()));
        if(c != null)
          return new StaticMethodExpr(source, line, tag, c, munge(sym.name), args);
        else
          return new InstanceMethodExpr(source, line, tag, instance, munge(sym.name), args);
        }
    }
  }

  private static Class maybeClass(Object form, boolean stringOk) throws Exception{
    if(form instanceof Class)
      return (Class) form;
    Class c = null;
    if(form instanceof Symbol)
      {
      Symbol sym = (Symbol) form;
      if(sym.ns == null) //if ns-qualified can't be classname
        {
        if(Util.equals(sym,COMPILE_STUB_SYM.get()))
          return (Class) COMPILE_STUB_CLASS.get();
        if(sym.name.indexOf('.') > 0 || sym.name.charAt(0) == '[')
          c = RT.classForName(sym.name);
        else
          {
          Object o = currentNS().getMapping(sym);
          if(o instanceof Class)
            c = (Class) o;
          }
        }
      }
    else if(stringOk && form instanceof String)
      c = RT.classForName((String) form);
    return c;
  }

  /*
   private static String maybeClassName(Object form, boolean stringOk){
     String className = null;
     if(form instanceof Symbol)
       {
       Symbol sym = (Symbol) form;
       if(sym.ns == null) //if ns-qualified can't be classname
         {
         if(sym.name.indexOf('.') > 0 || sym.name.charAt(0) == '[')
           className = sym.name;
         else
           {
           IPersistentMap imports = (IPersistentMap) ((Var) RT.NS_IMPORTS.get()).get();
           className = (String) imports.valAt(sym);
           }
         }
       }
     else if(stringOk && form instanceof String)
       className = (String) form;
     return className;
   }
*/
  static Class tagToClass(Object tag) throws Exception{
    Class c = maybeClass(tag, true);
    if(tag instanceof Symbol)
      {
      Symbol sym = (Symbol) tag;
      if(sym.ns == null) //if ns-qualified can't be classname
        {
        if(sym.name.equals("objects"))
          c = Object[].class;
        else if(sym.name.equals("ints"))
          c = int[].class;
        else if(sym.name.equals("longs"))
          c = long[].class;
        else if(sym.name.equals("floats"))
            c = float[].class;
          else if(sym.name.equals("doubles"))
              c = double[].class;
            else if(sym.name.equals("chars"))
                c = char[].class;
              else if(sym.name.equals("shorts"))
                  c = short[].class;
                else if(sym.name.equals("bytes"))
                    c = byte[].class;
                  else if(sym.name.equals("booleans"))
                      c = boolean[].class;
        }
      }
    if(c != null)
      return c;
    throw new IllegalArgumentException("Unable to resolve classname: " + tag);
  }
}

static abstract class FieldExpr extends HostExpr{
}

static class InstanceFieldExpr extends FieldExpr implements AssignableExpr{
  public final Expr target;
  public final Class targetClass;
  public final java.lang.reflect.Field field;
  public final String fieldName;
  public final int line;
  public final Symbol tag;
  final static Method invokeNoArgInstanceMember = Method.getMethod("Object invokeNoArgInstanceMember(Object,String)");
  final static Method setInstanceFieldMethod = Method.getMethod("Object setInstanceField(Object,String,Object)");


  public InstanceFieldExpr(int line, Expr target, String fieldName, Symbol tag) throws Exception{
    this.target = target;
    this.targetClass = target.hasJavaClass() ? target.getJavaClass() : null;
    this.field = targetClass != null ? Reflector.getField(targetClass, fieldName, false) : null;
    this.fieldName = fieldName;
    this.line = line;
    this.tag = tag;
    if(field == null && RT.booleanCast(RT.WARN_ON_REFLECTION.deref()))
      {
      RT.errPrintWriter()
          .format("Reflection warning, %s:%d - reference to field %s can't be resolved.\n",
            SOURCE_PATH.deref(), line, fieldName);
      }
  }

  public Object eval() throws Exception{
    return Reflector.invokeNoArgInstanceMember(target.eval(), fieldName);
  }

  public boolean canEmitPrimitive(){
    return targetClass != null && field != null &&
           Util.isPrimitive(field.getType());
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    if(targetClass != null && field != null)
      {
      target.emit(C.EXPRESSION, objx, gen);
      gen.checkCast(getType(targetClass));
      gen.getField(getType(targetClass), fieldName, Type.getType(field.getType()));
      }
    else
      throw new UnsupportedOperationException("Unboxed emit of unknown member");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    if(targetClass != null && field != null)
      {
      target.emit(C.EXPRESSION, objx, gen);
      gen.checkCast(getType(targetClass));
      gen.getField(getType(targetClass), fieldName, Type.getType(field.getType()));
      //if(context != C.STATEMENT)
      HostExpr.emitBoxReturn(objx, gen, field.getType());
      if(context == C.STATEMENT)
        {
        gen.pop();
        }
      }
    else
      {
      target.emit(C.EXPRESSION, objx, gen);
      gen.push(fieldName);
      gen.invokeStatic(REFLECTOR_TYPE, invokeNoArgInstanceMember);
      if(context == C.STATEMENT)
        gen.pop();
      }
  }

  public boolean hasJavaClass() throws Exception{
    return field != null || tag != null;
  }

  public Class getJavaClass() throws Exception{
    return tag != null ? HostExpr.tagToClass(tag) : field.getType();
  }

  public Object evalAssign(Expr val) throws Exception{
    return Reflector.setInstanceField(target.eval(), fieldName, val.eval());
  }

  public void emitAssign(C context, ObjExpr objx, GeneratorAdapter gen,
                         Expr val){
    gen.visitLineNumber(line, gen.mark());
    if(targetClass != null && field != null)
      {
      target.emit(C.EXPRESSION, objx, gen);
      gen.checkCast(Type.getType(targetClass));
      val.emit(C.EXPRESSION, objx, gen);
      gen.dupX1();
      HostExpr.emitUnboxArg(objx, gen, field.getType());
      gen.putField(Type.getType(targetClass), fieldName, Type.getType(field.getType()));
      }
    else
      {
      target.emit(C.EXPRESSION, objx, gen);
      gen.push(fieldName);
      val.emit(C.EXPRESSION, objx, gen);
      gen.invokeStatic(REFLECTOR_TYPE, setInstanceFieldMethod);
      }
    if(context == C.STATEMENT)
      gen.pop();
  }
}

static class StaticFieldExpr extends FieldExpr implements AssignableExpr{
  //final String className;
  public final String fieldName;
  public final Class c;
  public final java.lang.reflect.Field field;
  public final Symbol tag;
//  final static Method getStaticFieldMethod = Method.getMethod("Object getStaticField(String,String)");
//  final static Method setStaticFieldMethod = Method.getMethod("Object setStaticField(String,String,Object)");
  final int line;

  public StaticFieldExpr(int line, Class c, String fieldName, Symbol tag) throws Exception{
    //this.className = className;
    this.fieldName = fieldName;
    this.line = line;
    //c = Class.forName(className);
    this.c = c;
    field = c.getField(fieldName);
    this.tag = tag;
  }

  public Object eval() throws Exception{
    return Reflector.getStaticField(c, fieldName);
  }

  public boolean canEmitPrimitive(){
    return Util.isPrimitive(field.getType());
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    gen.getStatic(Type.getType(c), fieldName, Type.getType(field.getType()));
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());

    gen.getStatic(Type.getType(c), fieldName, Type.getType(field.getType()));
    //if(context != C.STATEMENT)
    HostExpr.emitBoxReturn(objx, gen, field.getType());
    if(context == C.STATEMENT)
      {
      gen.pop();
      }
//    gen.push(className);
//    gen.push(fieldName);
//    gen.invokeStatic(REFLECTOR_TYPE, getStaticFieldMethod);
  }

  public boolean hasJavaClass(){
    return true;
  }

  public Class getJavaClass() throws Exception{
    //Class c = Class.forName(className);
    //java.lang.reflect.Field field = c.getField(fieldName);
    return tag != null ? HostExpr.tagToClass(tag) : field.getType();
  }

  public Object evalAssign(Expr val) throws Exception{
    return Reflector.setStaticField(c, fieldName, val.eval());
  }

  public void emitAssign(C context, ObjExpr objx, GeneratorAdapter gen,
                         Expr val){
    gen.visitLineNumber(line, gen.mark());
    val.emit(C.EXPRESSION, objx, gen);
    gen.dup();
    HostExpr.emitUnboxArg(objx, gen, field.getType());
    gen.putStatic(Type.getType(c), fieldName, Type.getType(field.getType()));
    if(context == C.STATEMENT)
      gen.pop();
  }


}

static Class maybePrimitiveType(Expr e){
  try
    {
    if(e instanceof MaybePrimitiveExpr && e.hasJavaClass() && ((MaybePrimitiveExpr)e).canEmitPrimitive())
      {
      Class c = e.getJavaClass();
      if(Util.isPrimitive(c))
        return c;
      }
    }
  catch(Exception ex)
    {
    throw new RuntimeException(ex);
    }
  return null;
}

static abstract class MethodExpr extends HostExpr{
  static void emitArgsAsArray(IPersistentVector args, ObjExpr objx, GeneratorAdapter gen){
    gen.push(args.count());
    gen.newArray(OBJECT_TYPE);
    for(int i = 0; i < args.count(); i++)
      {
      gen.dup();
      gen.push(i);
      ((Expr) args.nth(i)).emit(C.EXPRESSION, objx, gen);
      gen.arrayStore(OBJECT_TYPE);
      }
  }

  public static void emitTypedArgs(ObjExpr objx, GeneratorAdapter gen, Class[] parameterTypes, IPersistentVector args){
    for(int i = 0; i < parameterTypes.length; i++)
      {
      Expr e = (Expr) args.nth(i);
      try
        {
        if(maybePrimitiveType(e) == parameterTypes[i])
          {
          ((MaybePrimitiveExpr) e).emitUnboxed(C.EXPRESSION, objx, gen);
          }
        else
          {
          e.emit(C.EXPRESSION, objx, gen);
          HostExpr.emitUnboxArg(objx, gen, parameterTypes[i]);
          }
        }
      catch(Exception e1)
        {
        e1.printStackTrace(RT.errPrintWriter());
        }

      }
  }
}

static class InstanceMethodExpr extends MethodExpr{
  public final Expr target;
  public final String methodName;
  public final IPersistentVector args;
  public final String source;
  public final int line;
  public final Symbol tag;
  public final java.lang.reflect.Method method;

  final static Method invokeInstanceMethodMethod =
      Method.getMethod("Object invokeInstanceMethod(Object,String,Object[])");


  public InstanceMethodExpr(String source, int line, Symbol tag, Expr target, String methodName, IPersistentVector args)
      throws Exception{
    this.source = source;
    this.line = line;
    this.args = args;
    this.methodName = methodName;
    this.target = target;
    this.tag = tag;
    if(target.hasJavaClass() && target.getJavaClass() != null)
      {
      List methods = Reflector.getMethods(target.getJavaClass(), args.count(), methodName, false);
      if(methods.isEmpty())
        method = null;
      //throw new IllegalArgumentException("No matching method found");
      else
        {
        int methodidx = 0;
        if(methods.size() > 1)
          {
          ArrayList<Class[]> params = new ArrayList();
          ArrayList<Class> rets = new ArrayList();
          for(int i = 0; i < methods.size(); i++)
            {
            java.lang.reflect.Method m = (java.lang.reflect.Method) methods.get(i);
            params.add(m.getParameterTypes());
            rets.add(m.getReturnType());
            }
          methodidx = getMatchingParams(methodName, params, args, rets);
          }
        java.lang.reflect.Method m =
            (java.lang.reflect.Method) (methodidx >= 0 ? methods.get(methodidx) : null);
        if(m != null && !Modifier.isPublic(m.getDeclaringClass().getModifiers()))
          {
          //public method of non-public class, try to find it in hierarchy
          m = Reflector.getAsMethodOfPublicBase(m.getDeclaringClass(), m);
          }
        method = m;
        }
      }
    else
      method = null;

    if(method == null && RT.booleanCast(RT.WARN_ON_REFLECTION.deref()))
      {
      RT.errPrintWriter()
          .format("Reflection warning, %s:%d - call to %s can't be resolved.\n",
            SOURCE_PATH.deref(), line, methodName);
      }
  }

  public Object eval() throws Exception{
    try
      {
      Object targetval = target.eval();
      Object[] argvals = new Object[args.count()];
      for(int i = 0; i < args.count(); i++)
        argvals[i] = ((Expr) args.nth(i)).eval();
      if(method != null)
        {
        LinkedList ms = new LinkedList();
        ms.add(method);
        return Reflector.invokeMatchingMethod(methodName, ms, targetval, argvals);
        }
      return Reflector.invokeInstanceMethod(targetval, methodName, argvals);
      }
    catch(Throwable e)
      {
      if(!(e instanceof CompilerException))
        throw new CompilerException(source, line, e);
      else
        throw (CompilerException) e;
      }
  }

  public boolean canEmitPrimitive(){
    return method != null && Util.isPrimitive(method.getReturnType());
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    if(method != null)
      {
      Type type = Type.getType(method.getDeclaringClass());
      target.emit(C.EXPRESSION, objx, gen);
      //if(!method.getDeclaringClass().isInterface())
      gen.checkCast(type);
      MethodExpr.emitTypedArgs(objx, gen, method.getParameterTypes(), args);
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      Method m = new Method(methodName, Type.getReturnType(method), Type.getArgumentTypes(method));
      if(method.getDeclaringClass().isInterface())
        gen.invokeInterface(type, m);
      else
        gen.invokeVirtual(type, m);
      }
    else
      throw new UnsupportedOperationException("Unboxed emit of unknown member");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    if(method != null)
      {
      Type type = Type.getType(method.getDeclaringClass());
      target.emit(C.EXPRESSION, objx, gen);
      //if(!method.getDeclaringClass().isInterface())
      gen.checkCast(type);
      MethodExpr.emitTypedArgs(objx, gen, method.getParameterTypes(), args);
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      Method m = new Method(methodName, Type.getReturnType(method), Type.getArgumentTypes(method));
      if(method.getDeclaringClass().isInterface())
        gen.invokeInterface(type, m);
      else
        gen.invokeVirtual(type, m);
      //if(context != C.STATEMENT || method.getReturnType() == Void.TYPE)
      HostExpr.emitBoxReturn(objx, gen, method.getReturnType());
      }
    else
      {
      target.emit(C.EXPRESSION, objx, gen);
      gen.push(methodName);
      emitArgsAsArray(args, objx, gen);
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      gen.invokeStatic(REFLECTOR_TYPE, invokeInstanceMethodMethod);
      }
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass(){
    return method != null || tag != null;
  }

  public Class getJavaClass() throws Exception{
    return tag != null ? HostExpr.tagToClass(tag) : method.getReturnType();
  }
}


static class StaticMethodExpr extends MethodExpr{
  //final String className;
  public final Class c;
  public final String methodName;
  public final IPersistentVector args;
  public final String source;
  public final int line;
  public final java.lang.reflect.Method method;
  public final Symbol tag;
  final static Method forNameMethod = Method.getMethod("Class forName(String)");
  final static Method invokeStaticMethodMethod =
      Method.getMethod("Object invokeStaticMethod(Class,String,Object[])");


  public StaticMethodExpr(String source, int line, Symbol tag, Class c, String methodName, IPersistentVector args)
      throws Exception{
    this.c = c;
    this.methodName = methodName;
    this.args = args;
    this.source = source;
    this.line = line;
    this.tag = tag;

    List methods = Reflector.getMethods(c, args.count(), methodName, true);
    if(methods.isEmpty())
      throw new IllegalArgumentException("No matching method: " + methodName);

    int methodidx = 0;
    if(methods.size() > 1)
      {
      ArrayList<Class[]> params = new ArrayList();
      ArrayList<Class> rets = new ArrayList();
      for(int i = 0; i < methods.size(); i++)
        {
        java.lang.reflect.Method m = (java.lang.reflect.Method) methods.get(i);
        params.add(m.getParameterTypes());
        rets.add(m.getReturnType());
        }
      methodidx = getMatchingParams(methodName, params, args, rets);
      }
    method = (java.lang.reflect.Method) (methodidx >= 0 ? methods.get(methodidx) : null);
    if(method == null && RT.booleanCast(RT.WARN_ON_REFLECTION.deref()))
      {
      RT.errPrintWriter()
              .format("Reflection warning, %s:%d - call to %s can't be resolved.\n",
                      SOURCE_PATH.deref(), line, methodName);
      }
  }

  public Object eval() throws Exception{
    try
      {
      Object[] argvals = new Object[args.count()];
      for(int i = 0; i < args.count(); i++)
        argvals[i] = ((Expr) args.nth(i)).eval();
      if(method != null)
        {
        LinkedList ms = new LinkedList();
        ms.add(method);
        return Reflector.invokeMatchingMethod(methodName, ms, null, argvals);
        }
      return Reflector.invokeStaticMethod(c, methodName, argvals);
      }
    catch(Throwable e)
      {
      if(!(e instanceof CompilerException))
        throw new CompilerException(source, line, e);
      else
        throw (CompilerException) e;
      }
  }

  public boolean canEmitPrimitive(){
    return method != null && Util.isPrimitive(method.getReturnType());
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    if(method != null)
      {
      MethodExpr.emitTypedArgs(objx, gen, method.getParameterTypes(), args);
      //Type type = Type.getObjectType(className.replace('.', '/'));
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      Type type = Type.getType(c);
      Method m = new Method(methodName, Type.getReturnType(method), Type.getArgumentTypes(method));
      gen.invokeStatic(type, m);
      }
    else
      throw new UnsupportedOperationException("Unboxed emit of unknown member");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    if(method != null)
      {
      MethodExpr.emitTypedArgs(objx, gen, method.getParameterTypes(), args);
      //Type type = Type.getObjectType(className.replace('.', '/'));
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      Type type = Type.getType(c);
      Method m = new Method(methodName, Type.getReturnType(method), Type.getArgumentTypes(method));
      gen.invokeStatic(type, m);
      //if(context != C.STATEMENT || method.getReturnType() == Void.TYPE)
      HostExpr.emitBoxReturn(objx, gen, method.getReturnType());
      }
    else
      {
      gen.push(c.getName());
      gen.invokeStatic(CLASS_TYPE, forNameMethod);
      gen.push(methodName);
      emitArgsAsArray(args, objx, gen);
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      gen.invokeStatic(REFLECTOR_TYPE, invokeStaticMethodMethod);
      }
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass(){
    return method != null || tag != null;
  }

  public Class getJavaClass() throws Exception{
    return tag != null ? HostExpr.tagToClass(tag) : method.getReturnType();
  }
}

static class UnresolvedVarExpr implements Expr{
  public final Symbol symbol;

  public UnresolvedVarExpr(Symbol symbol){
    this.symbol = symbol;
  }

  public boolean hasJavaClass(){
    return false;
  }

  public Class getJavaClass() throws Exception{
    throw new IllegalArgumentException(
        "UnresolvedVarExpr has no Java class");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
  }

  public Object eval() throws Exception{
    throw new IllegalArgumentException(
        "UnresolvedVarExpr cannot be evalled");
  }
}

static class ConstantExpr extends LiteralExpr{
  //stuff quoted vals in classloader at compile time, pull out at runtime
  //this won't work for static compilation...
  public final Object v;
  public final int id;

  public ConstantExpr(Object v){
    this.v = v;
    this.id = registerConstant(v);
//    this.id = RT.nextID();
//    DynamicClassLoader loader = (DynamicClassLoader) LOADER.get();
//    loader.registerQuotedVal(id, v);
  }

  Object val(){
    return v;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    objx.emitConstant(gen, id);
    if(context == C.STATEMENT)
      {
      gen.pop();
//      gen.loadThis();
//      gen.invokeVirtual(OBJECT_TYPE, getClassMethod);
//      gen.invokeVirtual(CLASS_TYPE, getClassLoaderMethod);
//      gen.checkCast(DYNAMIC_CLASSLOADER_TYPE);
//      gen.push(id);
//      gen.invokeVirtual(DYNAMIC_CLASSLOADER_TYPE, getQuotedValMethod);
      }
  }

  public boolean hasJavaClass(){
    return Modifier.isPublic(v.getClass().getModifiers());
    //return false;
  }

  public Class getJavaClass() throws Exception{
    return v.getClass();
    //throw new IllegalArgumentException("Has no Java class");
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object form){
      Object v = RT.second(form);

      if(v == null)
        return NIL_EXPR;
//      Class fclass = v.getClass();
//      if(fclass == Keyword.class)
//        return registerKeyword((Keyword) v);
//      else if(v instanceof Num)
//        return new NumExpr((Num) v);
//      else if(fclass == String.class)
//        return new StringExpr((String) v);
//      else if(fclass == Character.class)
//        return new CharExpr((Character) v);
//      else if(v instanceof IPersistentCollection && ((IPersistentCollection) v).count() == 0)
//        return new EmptyExpr(v);
      else
        return new ConstantExpr(v);
    }
  }
}

static class NilExpr extends LiteralExpr{
  Object val(){
    return null;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitInsn(Opcodes.ACONST_NULL);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass(){
    return true;
  }

  public Class getJavaClass() throws Exception{
    return null;
  }
}

final static NilExpr NIL_EXPR = new NilExpr();

static class BooleanExpr extends LiteralExpr{
  public final boolean val;


  public BooleanExpr(boolean val){
    this.val = val;
  }

  Object val(){
    return val ? RT.T : RT.F;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    if(val)
      gen.getStatic(BOOLEAN_OBJECT_TYPE, "TRUE", BOOLEAN_OBJECT_TYPE);
    else
      gen.getStatic(BOOLEAN_OBJECT_TYPE, "FALSE", BOOLEAN_OBJECT_TYPE);
    if(context == C.STATEMENT)
      {
      gen.pop();
      }
  }

  public boolean hasJavaClass(){
    return true;
  }

  public Class getJavaClass() throws Exception{
    return Boolean.class;
  }
}

final static BooleanExpr TRUE_EXPR = new BooleanExpr(true);
final static BooleanExpr FALSE_EXPR = new BooleanExpr(false);

static class StringExpr extends LiteralExpr{
  public final String str;

  public StringExpr(String str){
    this.str = str;
  }

  Object val(){
    return str;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    if(context != C.STATEMENT)
      gen.push(str);
  }

  public boolean hasJavaClass(){
    return true;
  }

  public Class getJavaClass() throws Exception{
    return String.class;
  }
}


static class MonitorEnterExpr extends UntypedExpr{
  final Expr target;

  public MonitorEnterExpr(Expr target){
    this.target = target;
  }

  public Object eval() throws Exception{
    throw new UnsupportedOperationException("Can't eval monitor-enter");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    target.emit(C.EXPRESSION, objx, gen);
    gen.monitorEnter();
    NIL_EXPR.emit(context, objx, gen);
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object form) throws Exception{
      return new MonitorEnterExpr(analyze(C.EXPRESSION, RT.second(form)));
    }
  }
}

static class MonitorExitExpr extends UntypedExpr{
  final Expr target;

  public MonitorExitExpr(Expr target){
    this.target = target;
  }

  public Object eval() throws Exception{
    throw new UnsupportedOperationException("Can't eval monitor-exit");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    target.emit(C.EXPRESSION, objx, gen);
    gen.monitorExit();
    NIL_EXPR.emit(context, objx, gen);
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object form) throws Exception{
      return new MonitorExitExpr(analyze(C.EXPRESSION, RT.second(form)));
    }
  }

}

public static class TryExpr implements Expr{
  public final Expr tryExpr;
  public final Expr finallyExpr;
  public final PersistentVector catchExprs;
  public final int retLocal;
  public final int finallyLocal;

  public static class CatchClause{
    //final String className;
    public final Class c;
    public final LocalBinding lb;
    public final Expr handler;
    Label label;
    Label endLabel;


    public CatchClause(Class c, LocalBinding lb, Expr handler){
      this.c = c;
      this.lb = lb;
      this.handler = handler;
    }
  }

  public TryExpr(Expr tryExpr, PersistentVector catchExprs, Expr finallyExpr, int retLocal, int finallyLocal){
    this.tryExpr = tryExpr;
    this.catchExprs = catchExprs;
    this.finallyExpr = finallyExpr;
    this.retLocal = retLocal;
    this.finallyLocal = finallyLocal;
  }

  public Object eval() throws Exception{
    throw new UnsupportedOperationException("Can't eval try");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    Label startTry = gen.newLabel();
    Label endTry = gen.newLabel();
    Label end = gen.newLabel();
    Label ret = gen.newLabel();
    Label finallyLabel = gen.newLabel();
    for(int i = 0; i < catchExprs.count(); i++)
      {
      CatchClause clause = (CatchClause) catchExprs.nth(i);
      clause.label = gen.newLabel();
      clause.endLabel = gen.newLabel();
      }

    gen.mark(startTry);
    tryExpr.emit(context, objx, gen);
    if(context != C.STATEMENT)
      gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), retLocal);
    gen.mark(endTry);
    if(finallyExpr != null)
      finallyExpr.emit(C.STATEMENT, objx, gen);
    gen.goTo(ret);

    for(int i = 0; i < catchExprs.count(); i++)
      {
      CatchClause clause = (CatchClause) catchExprs.nth(i);
      gen.mark(clause.label);
      //exception should be on stack
      //put in clause local
      gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), clause.lb.idx);
      clause.handler.emit(context, objx, gen);
      if(context != C.STATEMENT)
        gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), retLocal);
      gen.mark(clause.endLabel);

      if(finallyExpr != null)
        finallyExpr.emit(C.STATEMENT, objx, gen);
      gen.goTo(ret);
      }
    if(finallyExpr != null)
      {
      gen.mark(finallyLabel);
      //exception should be on stack
      gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), finallyLocal);
      finallyExpr.emit(C.STATEMENT, objx, gen);
      gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ILOAD), finallyLocal);
      gen.throwException();
      }
    gen.mark(ret);
    if(context != C.STATEMENT)
      gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ILOAD), retLocal);
    gen.mark(end);
    for(int i = 0; i < catchExprs.count(); i++)
      {
      CatchClause clause = (CatchClause) catchExprs.nth(i);
      gen.visitTryCatchBlock(startTry, endTry, clause.label, clause.c.getName().replace('.', '/'));
      }
    if(finallyExpr != null)
      {
        gen.visitTryCatchBlock(startTry, endTry, finallyLabel, null);
        for(int i = 0; i < catchExprs.count(); i++)
          {
          CatchClause clause = (CatchClause) catchExprs.nth(i);
          gen.visitTryCatchBlock(clause.label, clause.endLabel, finallyLabel, null);
          }
      }
    for(int i = 0; i < catchExprs.count(); i++)
      {
      CatchClause clause = (CatchClause) catchExprs.nth(i);
      gen.visitLocalVariable(clause.lb.name, "Ljava/lang/Object;", null, clause.label, clause.endLabel,
                             clause.lb.idx);
      }
  }

  public boolean hasJavaClass() throws Exception{
    return tryExpr.hasJavaClass();
  }

  public Class getJavaClass() throws Exception{
    return tryExpr.getJavaClass();
  }

  static class Parser implements IParser{

    public Expr parse(C context, Object frm) throws Exception{
      ISeq form = (ISeq) frm;
//      if(context == C.EVAL || context == C.EXPRESSION)
      if(context != C.RETURN)
        return analyze(context, RT.list(RT.list(FN, PersistentVector.EMPTY, form)));

      //(try try-expr* catch-expr* finally-expr?)
      //catch-expr: (catch class sym expr*)
      //finally-expr: (finally expr*)

      PersistentVector body = PersistentVector.EMPTY;
      PersistentVector catches = PersistentVector.EMPTY;
            Expr bodyExpr = null;
      Expr finallyExpr = null;
      boolean caught = false;

      int retLocal = getAndIncLocalNum();
      int finallyLocal = getAndIncLocalNum();
      for(ISeq fs = form.next(); fs != null; fs = fs.next())
        {
        Object f = fs.first();
        Object op = (f instanceof ISeq) ? ((ISeq) f).first() : null;
        if(!Util.equals(op, CATCH) && !Util.equals(op, FINALLY))
          {
          if(caught)
            throw new Exception("Only catch or finally clause can follow catch in try expression");
          body = body.cons(f);
          }
        else
          {
                    if(bodyExpr == null)
                        bodyExpr = (new BodyExpr.Parser()).parse(context, RT.seq(body));
          if(Util.equals(op, CATCH))
            {
            Class c = HostExpr.maybeClass(RT.second(f), false);
            if(c == null)
              throw new IllegalArgumentException("Unable to resolve classname: " + RT.second(f));
            if(!(RT.third(f) instanceof Symbol))
              throw new IllegalArgumentException(
                  "Bad binding form, expected symbol, got: " + RT.third(f));
            Symbol sym = (Symbol) RT.third(f);
            if(sym.getNamespace() != null)
              throw new Exception("Can't bind qualified name:" + sym);

            IPersistentMap dynamicBindings = RT.map(LOCAL_ENV, LOCAL_ENV.deref(),
                                                    NEXT_LOCAL_NUM, NEXT_LOCAL_NUM.deref(),
                                                    IN_CATCH_FINALLY, RT.T);
            try
              {
              Var.pushThreadBindings(dynamicBindings);
              LocalBinding lb = registerLocal(sym,
                                              (Symbol) (RT.second(f) instanceof Symbol ? RT.second(f)
                                                                                       : null),
                                              null,false);
              Expr handler = (new BodyExpr.Parser()).parse(context, RT.next(RT.next(RT.next(f))));
              catches = catches.cons(new CatchClause(c, lb, handler));
              }
            finally
              {
              Var.popThreadBindings();
              }
            caught = true;
            }
          else //finally
            {
            if(fs.next() != null)
              throw new Exception("finally clause must be last in try expression");
            try
              {
              Var.pushThreadBindings(RT.map(IN_CATCH_FINALLY, RT.T));
              finallyExpr = (new BodyExpr.Parser()).parse(C.STATEMENT, RT.next(f));
              }
            finally
              {
              Var.popThreadBindings();
              }
            }
          }
        }
            if(bodyExpr == null)
                bodyExpr = (new BodyExpr.Parser()).parse(context, RT.seq(body));

      return new TryExpr(bodyExpr, catches, finallyExpr, retLocal,
                         finallyLocal);
    }
  }
}

//static class TryFinallyExpr implements Expr{
//  final Expr tryExpr;
//  final Expr finallyExpr;
//
//
//  public TryFinallyExpr(Expr tryExpr, Expr finallyExpr){
//    this.tryExpr = tryExpr;
//    this.finallyExpr = finallyExpr;
//  }
//
//  public Object eval() throws Exception{
//    throw new UnsupportedOperationException("Can't eval try");
//  }
//
//  public void emit(C context, FnExpr fn, GeneratorAdapter gen){
//    Label startTry = gen.newLabel();
//    Label endTry = gen.newLabel();
//    Label end = gen.newLabel();
//    Label finallyLabel = gen.newLabel();
//    gen.visitTryCatchBlock(startTry, endTry, finallyLabel, null);
//    gen.mark(startTry);
//    tryExpr.emit(context, fn, gen);
//    gen.mark(endTry);
//    finallyExpr.emit(C.STATEMENT, fn, gen);
//    gen.goTo(end);
//    gen.mark(finallyLabel);
//    //exception should be on stack
//    finallyExpr.emit(C.STATEMENT, fn, gen);
//    gen.throwException();
//    gen.mark(end);
//  }
//
//  public boolean hasJavaClass() throws Exception{
//    return tryExpr.hasJavaClass();
//  }
//
//  public Class getJavaClass() throws Exception{
//    return tryExpr.getJavaClass();
//  }
//
//  static class Parser implements IParser{
//    public Expr parse(C context, Object frm) throws Exception{
//      ISeq form = (ISeq) frm;
//      //(try-finally try-expr finally-expr)
//      if(form.count() != 3)
//        throw new IllegalArgumentException(
//            "Wrong number of arguments, expecting: (try-finally try-expr finally-expr) ");
//
//      if(context == C.EVAL || context == C.EXPRESSION)
//        return analyze(context, RT.list(RT.list(FN, PersistentVector.EMPTY, form)));
//
//      return new TryFinallyExpr(analyze(context, RT.second(form)),
//                                analyze(C.STATEMENT, RT.third(form)));
//    }
//  }
//}

static class ThrowExpr extends UntypedExpr{
  public final Expr excExpr;

  public ThrowExpr(Expr excExpr){
    this.excExpr = excExpr;
  }


  public Object eval() throws Exception{
    throw new Exception("Can't eval throw");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    excExpr.emit(C.EXPRESSION, objx, gen);
    gen.checkCast(THROWABLE_TYPE);
    gen.throwException();
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object form) throws Exception{
      if(context == C.EVAL)
        return analyze(context, RT.list(RT.list(FN, PersistentVector.EMPTY, form)));
      return new ThrowExpr(analyze(C.EXPRESSION, RT.second(form)));
    }
  }
}


static public boolean subsumes(Class[] c1, Class[] c2){
  //presumes matching lengths
  Boolean better = false;
  for(int i = 0; i < c1.length; i++)
    {
    if(c1[i] != c2[i])// || c2[i].isPrimitive() && c1[i] == Object.class))
      {
      if(!c1[i].isPrimitive() && c2[i].isPrimitive()
         //|| Number.class.isAssignableFrom(c1[i]) && c2[i].isPrimitive()
         ||
         c2[i].isAssignableFrom(c1[i]))
        better = true;
      else
        return false;
      }
    }
  return better;
}

static int getMatchingParams(String methodName, ArrayList<Class[]> paramlists, IPersistentVector argexprs,
                             List<Class> rets)
    throws Exception{
  //presumes matching lengths
  int matchIdx = -1;
  boolean tied = false;
    boolean foundExact = false;
  for(int i = 0; i < paramlists.size(); i++)
    {
    boolean match = true;
    ISeq aseq = argexprs.seq();
    int exact = 0;
    for(int p = 0; match && p < argexprs.count() && aseq != null; ++p, aseq = aseq.next())
      {
      Expr arg = (Expr) aseq.first();
      Class aclass = arg.hasJavaClass() ? arg.getJavaClass() : Object.class;
      Class pclass = paramlists.get(i)[p];
      if(arg.hasJavaClass() && aclass == pclass)
        exact++;
      else
        match = Reflector.paramArgTypeMatch(pclass, aclass);
      }
    if(exact == argexprs.count())
            {
            if(!foundExact || matchIdx == -1 || rets.get(matchIdx).isAssignableFrom(rets.get(i)))
                matchIdx = i;
            foundExact = true;
            }
    else if(match && !foundExact)
      {
      if(matchIdx == -1)
        matchIdx = i;
      else
        {
        if(subsumes(paramlists.get(i), paramlists.get(matchIdx)))
          {
          matchIdx = i;
          tied = false;
          }
        else if(Arrays.equals(paramlists.get(matchIdx), paramlists.get(i)))
          {
          if(rets.get(matchIdx).isAssignableFrom(rets.get(i)))
            matchIdx = i;
          }
        else if(!(subsumes(paramlists.get(matchIdx), paramlists.get(i))))
            tied = true;
        }
      }
    }
  if(tied)
    throw new IllegalArgumentException("More than one matching method found: " + methodName);

  return matchIdx;
}

public static class NewExpr implements Expr{
  public final IPersistentVector args;
  public final Constructor ctor;
  public final Class c;
  final static Method invokeConstructorMethod =
      Method.getMethod("Object invokeConstructor(Class,Object[])");
//  final static Method forNameMethod = Method.getMethod("Class classForName(String)");
  final static Method forNameMethod = Method.getMethod("Class forName(String)");


  public NewExpr(Class c, IPersistentVector args, int line) throws Exception{
    this.args = args;
    this.c = c;
    Constructor[] allctors = c.getConstructors();
    ArrayList ctors = new ArrayList();
    ArrayList<Class[]> params = new ArrayList();
    ArrayList<Class> rets = new ArrayList();
    for(int i = 0; i < allctors.length; i++)
      {
      Constructor ctor = allctors[i];
      if(ctor.getParameterTypes().length == args.count())
        {
        ctors.add(ctor);
        params.add(ctor.getParameterTypes());
        rets.add(c);
        }
      }
    if(ctors.isEmpty())
      throw new IllegalArgumentException("No matching ctor found for " + c);

    int ctoridx = 0;
    if(ctors.size() > 1)
      {
      ctoridx = getMatchingParams(c.getName(), params, args, rets);
      }

    this.ctor = ctoridx >= 0 ? (Constructor) ctors.get(ctoridx) : null;
    if(ctor == null && RT.booleanCast(RT.WARN_ON_REFLECTION.deref()))
      {
      RT.errPrintWriter()
              .format("Reflection warning, %s:%d - call to %s ctor can't be resolved.\n",
                      SOURCE_PATH.deref(), line, c.getName());
      }
  }

  public Object eval() throws Exception{
    Object[] argvals = new Object[args.count()];
    for(int i = 0; i < args.count(); i++)
      argvals[i] = ((Expr) args.nth(i)).eval();
    if(this.ctor != null)
      {
      return ctor.newInstance(Reflector.boxArgs(ctor.getParameterTypes(), argvals));
      }
    return Reflector.invokeConstructor(c, argvals);
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    if(this.ctor != null)
      {
      Type type = getType(c);
      gen.newInstance(type);
      gen.dup();
      MethodExpr.emitTypedArgs(objx, gen, ctor.getParameterTypes(), args);
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      gen.invokeConstructor(type, new Method("<init>", Type.getConstructorDescriptor(ctor)));
      }
    else
      {
      gen.push(destubClassName(c.getName()));
      gen.invokeStatic(CLASS_TYPE, forNameMethod);
      MethodExpr.emitArgsAsArray(args, objx, gen);
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      gen.invokeStatic(REFLECTOR_TYPE, invokeConstructorMethod);
      }
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass(){
    return true;
  }

  public Class getJavaClass() throws Exception{
    return c;
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object frm) throws Exception{
      int line = (Integer) LINE.deref();
      ISeq form = (ISeq) frm;
      //(new Classname args...)
      if(form.count() < 2)
        throw new Exception("wrong number of arguments, expecting: (new Classname args...)");
      Class c = HostExpr.maybeClass(RT.second(form), false);
      if(c == null)
        throw new IllegalArgumentException("Unable to resolve classname: " + RT.second(form));
      PersistentVector args = PersistentVector.EMPTY;
      for(ISeq s = RT.next(RT.next(form)); s != null; s = s.next())
        args = args.cons(analyze(context == C.EVAL ? context : C.EXPRESSION, s.first()));
      return new NewExpr(c, args, line);
    }
  }

}

public static class MetaExpr implements Expr{
  public final Expr expr;
  public final MapExpr meta;
  final static Type IOBJ_TYPE = Type.getType(IObj.class);
  final static Method withMetaMethod = Method.getMethod("clojure.lang.IObj withMeta(clojure.lang.IPersistentMap)");


  public MetaExpr(Expr expr, MapExpr meta){
    this.expr = expr;
    this.meta = meta;
  }

  public Object eval() throws Exception{
    return ((IObj) expr.eval()).withMeta((IPersistentMap) meta.eval());
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    expr.emit(C.EXPRESSION, objx, gen);
    gen.checkCast(IOBJ_TYPE);
    meta.emit(C.EXPRESSION, objx, gen);
    gen.checkCast(IPERSISTENTMAP_TYPE);
    gen.invokeInterface(IOBJ_TYPE, withMetaMethod);
    if(context == C.STATEMENT)
      {
      gen.pop();
      }
  }

  public boolean hasJavaClass() throws Exception{
    return expr.hasJavaClass();
  }

  public Class getJavaClass() throws Exception{
    return expr.getJavaClass();
  }
}

public static class IfExpr implements Expr, MaybePrimitiveExpr{
  public final Expr testExpr;
  public final Expr thenExpr;
  public final Expr elseExpr;
  public final int line;


  public IfExpr(int line, Expr testExpr, Expr thenExpr, Expr elseExpr){
    this.testExpr = testExpr;
    this.thenExpr = thenExpr;
    this.elseExpr = elseExpr;
    this.line = line;
  }

  public Object eval() throws Exception{
    Object t = testExpr.eval();
    if(t != null && t != Boolean.FALSE)
      return thenExpr.eval();
    return elseExpr.eval();
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    doEmit(context, objx, gen,false);
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    doEmit(context, objx, gen, true);
  }

  public void doEmit(C context, ObjExpr objx, GeneratorAdapter gen, boolean emitUnboxed){
    Label nullLabel = gen.newLabel();
    Label falseLabel = gen.newLabel();
    Label endLabel = gen.newLabel();

    gen.visitLineNumber(line, gen.mark());

    try
      {
      if(maybePrimitiveType(testExpr) == boolean.class)
        {
        ((MaybePrimitiveExpr) testExpr).emitUnboxed(C.EXPRESSION, objx, gen);
        gen.ifZCmp(gen.EQ, falseLabel);
        }
      else
        {
        testExpr.emit(C.EXPRESSION, objx, gen);
        gen.dup();
        gen.ifNull(nullLabel);
        gen.getStatic(BOOLEAN_OBJECT_TYPE, "FALSE", BOOLEAN_OBJECT_TYPE);
        gen.visitJumpInsn(IF_ACMPEQ, falseLabel);
        }
      }
    catch(Exception e)
      {
      throw new RuntimeException(e);
      }
    if(emitUnboxed)
      ((MaybePrimitiveExpr)thenExpr).emitUnboxed(context, objx, gen);
    else
      thenExpr.emit(context, objx, gen);
    gen.goTo(endLabel);
    gen.mark(nullLabel);
    gen.pop();
    gen.mark(falseLabel);
    if(emitUnboxed)
      ((MaybePrimitiveExpr)elseExpr).emitUnboxed(context, objx, gen);
    else
      elseExpr.emit(context, objx, gen);
    gen.mark(endLabel);
  }

  public boolean hasJavaClass() throws Exception{
    return thenExpr.hasJavaClass()
           && elseExpr.hasJavaClass()
           &&
           (thenExpr.getJavaClass() == elseExpr.getJavaClass()
            || (thenExpr.getJavaClass() == null && !elseExpr.getJavaClass().isPrimitive())
            || (elseExpr.getJavaClass() == null && !thenExpr.getJavaClass().isPrimitive()));
  }

  public boolean canEmitPrimitive(){
    try
      {
      return thenExpr instanceof MaybePrimitiveExpr
             && elseExpr instanceof MaybePrimitiveExpr
             && thenExpr.getJavaClass() == elseExpr.getJavaClass()
             && ((MaybePrimitiveExpr)thenExpr).canEmitPrimitive()
           && ((MaybePrimitiveExpr)elseExpr).canEmitPrimitive();
      }
    catch(Exception e)
      {
      return false;
      }
  }

  public Class getJavaClass() throws Exception{
    Class thenClass = thenExpr.getJavaClass();
    if(thenClass != null)
      return thenClass;
    return elseExpr.getJavaClass();
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object frm) throws Exception{
      ISeq form = (ISeq) frm;
      //(if test then) or (if test then else)
      if(form.count() > 4)
        throw new Exception("Too many arguments to if");
      else if(form.count() < 3)
        throw new Exception("Too few arguments to if");
            PathNode branch = new PathNode(PATHTYPE.BRANCH, (PathNode) CLEAR_PATH.get());
            Expr testexpr = analyze(context == C.EVAL ? context : C.EXPRESSION, RT.second(form));
            Expr thenexpr, elseexpr;
            try {
                Var.pushThreadBindings(
                        RT.map(CLEAR_PATH, new PathNode(PATHTYPE.PATH,branch)));
                thenexpr = analyze(context, RT.third(form));
                }
            finally{
                Var.popThreadBindings();
                }
            try {
                Var.pushThreadBindings(
                        RT.map(CLEAR_PATH, new PathNode(PATHTYPE.PATH,branch)));
                elseexpr = analyze(context, RT.fourth(form));
                }
            finally{
                Var.popThreadBindings();
                }
      return new IfExpr((Integer) LINE.deref(),
                        testexpr,
                        thenexpr,
                        elseexpr);
    }
  }
}

static final public IPersistentMap CHAR_MAP =
    PersistentHashMap.create('-', "_",
//                             '.', "_DOT_",
':', "_COLON_",
'+', "_PLUS_",
'>', "_GT_",
'<', "_LT_",
'=', "_EQ_",
'~', "_TILDE_",
'!', "_BANG_",
'@', "_CIRCA_",
'#', "_SHARP_",
'$', "_DOLLARSIGN_",
'%', "_PERCENT_",
'^', "_CARET_",
'&', "_AMPERSAND_",
'*', "_STAR_",
'|', "_BAR_",
'{', "_LBRACE_",
'}', "_RBRACE_",
'[', "_LBRACK_",
']', "_RBRACK_",
'/', "_SLASH_",
'\\', "_BSLASH_",
'?', "_QMARK_");

static public String munge(String name){
  StringBuilder sb = new StringBuilder();
  for(char c : name.toCharArray())
    {
    String sub = (String) CHAR_MAP.valAt(c);
    if(sub != null)
      sb.append(sub);
    else
      sb.append(c);
    }
  return sb.toString();
}

public static class EmptyExpr implements Expr{
  public final Object coll;
  final static Type HASHMAP_TYPE = Type.getType(PersistentArrayMap.class);
  final static Type HASHSET_TYPE = Type.getType(PersistentHashSet.class);
  final static Type VECTOR_TYPE = Type.getType(PersistentVector.class);
  final static Type LIST_TYPE = Type.getType(PersistentList.class);
  final static Type EMPTY_LIST_TYPE = Type.getType(PersistentList.EmptyList.class);


  public EmptyExpr(Object coll){
    this.coll = coll;
  }

  public Object eval() throws Exception{
    return coll;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    if(coll instanceof IPersistentList)
      gen.getStatic(LIST_TYPE, "EMPTY", EMPTY_LIST_TYPE);
    else if(coll instanceof IPersistentVector)
      gen.getStatic(VECTOR_TYPE, "EMPTY", VECTOR_TYPE);
    else if(coll instanceof IPersistentMap)
        gen.getStatic(HASHMAP_TYPE, "EMPTY", HASHMAP_TYPE);
      else if(coll instanceof IPersistentSet)
          gen.getStatic(HASHSET_TYPE, "EMPTY", HASHSET_TYPE);
        else
          throw new UnsupportedOperationException("Unknown Collection type");
    if(context == C.STATEMENT)
      {
      gen.pop();
      }
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    if(coll instanceof IPersistentList)
      return IPersistentList.class;
    else if(coll instanceof IPersistentVector)
      return IPersistentVector.class;
    else if(coll instanceof IPersistentMap)
        return IPersistentMap.class;
      else if(coll instanceof IPersistentSet)
          return IPersistentSet.class;
        else
          throw new UnsupportedOperationException("Unknown Collection type");
  }
}

public static class ListExpr implements Expr{
  public final IPersistentVector args;
  final static Method arrayToListMethod = Method.getMethod("clojure.lang.ISeq arrayToList(Object[])");


  public ListExpr(IPersistentVector args){
    this.args = args;
  }

  public Object eval() throws Exception{
    IPersistentVector ret = PersistentVector.EMPTY;
    for(int i = 0; i < args.count(); i++)
      ret = (IPersistentVector) ret.cons(((Expr) args.nth(i)).eval());
    return ret.seq();
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    MethodExpr.emitArgsAsArray(args, objx, gen);
    gen.invokeStatic(RT_TYPE, arrayToListMethod);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    return IPersistentList.class;
  }

}

public static class MapExpr implements Expr{
  public final IPersistentVector keyvals;
  final static Method mapMethod = Method.getMethod("clojure.lang.IPersistentMap map(Object[])");


  public MapExpr(IPersistentVector keyvals){
    this.keyvals = keyvals;
  }

  public Object eval() throws Exception{
    Object[] ret = new Object[keyvals.count()];
    for(int i = 0; i < keyvals.count(); i++)
      ret[i] = ((Expr) keyvals.nth(i)).eval();
    return RT.map(ret);
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    MethodExpr.emitArgsAsArray(keyvals, objx, gen);
    gen.invokeStatic(RT_TYPE, mapMethod);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    return IPersistentMap.class;
  }


  static public Expr parse(C context, IPersistentMap form) throws Exception{
    IPersistentVector keyvals = PersistentVector.EMPTY;
    for(ISeq s = RT.seq(form); s != null; s = s.next())
      {
      IMapEntry e = (IMapEntry) s.first();
      keyvals = (IPersistentVector) keyvals.cons(analyze(context == C.EVAL ? context : C.EXPRESSION, e.key()));
      keyvals = (IPersistentVector) keyvals.cons(analyze(context == C.EVAL ? context : C.EXPRESSION, e.val()));
      }
    Expr ret = new MapExpr(keyvals);
    if(form instanceof IObj && ((IObj) form).meta() != null)
      return new MetaExpr(ret, (MapExpr) MapExpr
          .parse(context == C.EVAL ? context : C.EXPRESSION, ((IObj) form).meta()));
    else
      return ret;
  }
}

public static class SetExpr implements Expr{
  public final IPersistentVector keys;
  final static Method setMethod = Method.getMethod("clojure.lang.IPersistentSet set(Object[])");


  public SetExpr(IPersistentVector keys){
    this.keys = keys;
  }

  public Object eval() throws Exception{
    Object[] ret = new Object[keys.count()];
    for(int i = 0; i < keys.count(); i++)
      ret[i] = ((Expr) keys.nth(i)).eval();
    return RT.set(ret);
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    MethodExpr.emitArgsAsArray(keys, objx, gen);
    gen.invokeStatic(RT_TYPE, setMethod);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    return IPersistentSet.class;
  }


  static public Expr parse(C context, IPersistentSet form) throws Exception{
    IPersistentVector keys = PersistentVector.EMPTY;
    for(ISeq s = RT.seq(form); s != null; s = s.next())
      {
      Object e = s.first();
      keys = (IPersistentVector) keys.cons(analyze(context == C.EVAL ? context : C.EXPRESSION, e));
      }
    Expr ret = new SetExpr(keys);
    if(form instanceof IObj && ((IObj) form).meta() != null)
      return new MetaExpr(ret, (MapExpr) MapExpr
          .parse(context == C.EVAL ? context : C.EXPRESSION, ((IObj) form).meta()));
    else
      return ret;
  }
}

public static class VectorExpr implements Expr{
  public final IPersistentVector args;
  final static Method vectorMethod = Method.getMethod("clojure.lang.IPersistentVector vector(Object[])");


  public VectorExpr(IPersistentVector args){
    this.args = args;
  }

  public Object eval() throws Exception{
    IPersistentVector ret = PersistentVector.EMPTY;
    for(int i = 0; i < args.count(); i++)
      ret = (IPersistentVector) ret.cons(((Expr) args.nth(i)).eval());
    return ret;
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    MethodExpr.emitArgsAsArray(args, objx, gen);
    gen.invokeStatic(RT_TYPE, vectorMethod);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    return IPersistentVector.class;
  }

  static public Expr parse(C context, IPersistentVector form) throws Exception{
    IPersistentVector args = PersistentVector.EMPTY;
    for(int i = 0; i < form.count(); i++)
      args = (IPersistentVector) args.cons(analyze(context == C.EVAL ? context : C.EXPRESSION, form.nth(i)));
    Expr ret = new VectorExpr(args);
    if(form instanceof IObj && ((IObj) form).meta() != null)
      return new MetaExpr(ret, (MapExpr) MapExpr
          .parse(context == C.EVAL ? context : C.EXPRESSION, ((IObj) form).meta()));
    else
      return ret;
  }

}

static class KeywordInvokeExpr implements Expr{
  public final KeywordExpr kw;
  public final Object tag;
  public final Expr target;
  public final int line;
  public final int siteIndex;
  public final String source;
  static Type ILOOKUP_TYPE = Type.getType(ILookup.class);

  public KeywordInvokeExpr(String source, int line, Symbol tag, KeywordExpr kw, Expr target){
    this.source = source;
    this.kw = kw;
    this.target = target;
    this.line = line;
    this.tag = tag;
    this.siteIndex = registerKeywordCallsite(kw.k);
  }

  public Object eval() throws Exception{
    try
      {
      return kw.k.invoke(target.eval());
      }
    catch(Throwable e)
      {
      if(!(e instanceof CompilerException))
        throw new CompilerException(source, line, e);
      else
        throw (CompilerException) e;
      }
  }

    public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
        Label endLabel = gen.newLabel();
        Label faultLabel = gen.newLabel();

        gen.visitLineNumber(line, gen.mark());
        gen.getStatic(objx.objtype, objx.thunkNameStatic(siteIndex),ObjExpr.ILOOKUP_THUNK_TYPE);
        gen.dup();
        target.emit(C.EXPRESSION, objx, gen);
        gen.dupX2();
        gen.invokeInterface(ObjExpr.ILOOKUP_THUNK_TYPE, Method.getMethod("Object get(Object)"));
        gen.dupX2();
        gen.visitJumpInsn(IF_ACMPEQ, faultLabel);
        gen.pop();
        gen.goTo(endLabel);

        gen.mark(faultLabel);
        gen.swap();
        gen.pop();
        gen.getStatic(objx.objtype, objx.siteNameStatic(siteIndex),ObjExpr.KEYWORD_LOOKUPSITE_TYPE);
        gen.swap();
        gen.loadThis();
        gen.invokeInterface(ObjExpr.ILOOKUP_SITE_TYPE,
                            Method.getMethod("Object fault(Object, clojure.lang.ILookupHost)"));

        gen.mark(endLabel);
        if(context == C.STATEMENT)
            gen.pop();
    }

  public void emit2(C context, ObjExpr objx, GeneratorAdapter gen){
    Label endLabel = gen.newLabel();
    Label faultLabel = gen.newLabel();

    gen.visitLineNumber(line, gen.mark());
    target.emit(C.EXPRESSION, objx, gen);
    gen.dup();
    gen.getStatic(objx.objtype, objx.thunkNameStatic(siteIndex),ObjExpr.ILOOKUP_THUNK_TYPE);
    gen.swap();
    gen.getStatic(objx.objtype, objx.siteNameStatic(siteIndex),ObjExpr.KEYWORD_LOOKUPSITE_TYPE);
///    gen.loadThis();
    gen.invokeInterface(ObjExpr.ILOOKUP_THUNK_TYPE,
                        Method.getMethod("Object get(Object,clojure.lang.ILookupSite)"));
//    gen.invokeInterface(ObjExpr.ILOOKUP_THUNK_TYPE,
//                        Method.getMethod("Object get(Object,clojure.lang.ILookupSite,clojure.lang.ILookupHost)"));
    gen.dup();
    gen.getStatic(objx.objtype, objx.siteNameStatic(siteIndex),ObjExpr.KEYWORD_LOOKUPSITE_TYPE);
    gen.visitJumpInsn(IF_ACMPEQ, faultLabel);
    gen.swap();
    gen.pop();
    gen.goTo(endLabel);

    gen.mark(faultLabel);
    gen.swap();
    gen.loadThis();
    gen.invokeInterface(ObjExpr.ILOOKUP_SITE_TYPE,
                        Method.getMethod("Object fault(Object, clojure.lang.ILookupHost)"));
       
    gen.mark(endLabel);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public void emitInstance(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    gen.loadThis();
    gen.getField(objx.objtype, objx.thunkName(siteIndex),ObjExpr.ILOOKUP_THUNK_TYPE);
    target.emit(C.EXPRESSION, objx, gen);
    gen.loadThis();
    gen.getField(objx.objtype, objx.siteName(siteIndex),ObjExpr.ILOOKUP_SITE_TYPE);
    gen.loadThis();
    gen.checkCast(Type.getType(ILookupHost.class));
    gen.invokeInterface(ObjExpr.ILOOKUP_THUNK_TYPE,
                        Method.getMethod("Object get(Object,clojure.lang.ILookupSite,clojure.lang.ILookupHost)"));
    if(context == C.STATEMENT)
      gen.pop();
  }

  public void emitNormal(C context, ObjExpr objx, GeneratorAdapter gen){
    Label slowLabel = gen.newLabel();
    Label endLabel = gen.newLabel();

    gen.visitLineNumber(line, gen.mark());
    target.emit(C.EXPRESSION, objx, gen);
    gen.dup();
    gen.instanceOf(ILOOKUP_TYPE);
    gen.ifZCmp(GeneratorAdapter.EQ, slowLabel);
    kw.emit(C.EXPRESSION, objx, gen);
    gen.invokeInterface(ILOOKUP_TYPE, new Method("valAt", OBJECT_TYPE, ARG_TYPES[1]));
    gen.goTo(endLabel);

    gen.mark(slowLabel);
    kw.emit(C.EXPRESSION, objx, gen);
    gen.invokeStatic(RT_TYPE, new Method("get", OBJECT_TYPE, ARG_TYPES[2]));

    gen.mark(endLabel);

    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass() throws Exception{
    return tag != null;
  }

  public Class getJavaClass() throws Exception{
    return HostExpr.tagToClass(tag);
  }

}
//static class KeywordSiteInvokeExpr implements Expr{
//  public final Expr site;
//  public final Object tag;
//  public final Expr target;
//  public final int line;
//  public final String source;
//
//  public KeywordSiteInvokeExpr(String source, int line, Symbol tag, Expr site, Expr target){
//    this.source = source;
//    this.site = site;
//    this.target = target;
//    this.line = line;
//    this.tag = tag;
//  }
//
//  public Object eval() throws Exception{
//    try
//      {
//      KeywordCallSite s = (KeywordCallSite) site.eval();
//      return s.thunk.invoke(s,target.eval());
//      }
//    catch(Throwable e)
//      {
//      if(!(e instanceof CompilerException))
//        throw new CompilerException(source, line, e);
//      else
//        throw (CompilerException) e;
//      }
//  }
//
//  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
//    gen.visitLineNumber(line, gen.mark());
//    site.emit(C.EXPRESSION, objx, gen);
//    gen.dup();
//    gen.getField(Type.getType(KeywordCallSite.class),"thunk",IFN_TYPE);
//    gen.swap();
//    target.emit(C.EXPRESSION, objx, gen);
//
//    gen.invokeInterface(IFN_TYPE, new Method("invoke", OBJECT_TYPE, ARG_TYPES[2]));
//    if(context == C.STATEMENT)
//      gen.pop();
//  }
//
//  public boolean hasJavaClass() throws Exception{
//    return tag != null;
//  }
//
//  public Class getJavaClass() throws Exception{
//    return HostExpr.tagToClass(tag);
//  }
//
//}

public static class InstanceOfExpr implements Expr, MaybePrimitiveExpr{
  Expr expr;
  Class c;

  public InstanceOfExpr(Class c, Expr expr){
    this.expr = expr;
    this.c = c;
  }

  public Object eval() throws Exception{
    if(c.isInstance(expr.eval()))
      return RT.T;
    return RT.F;
  }

  public boolean canEmitPrimitive(){
    return true;
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    expr.emit(C.EXPRESSION,objx,gen);
    gen.instanceOf(Type.getType(c));
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    emitUnboxed(context,objx,gen);
    HostExpr.emitBoxReturn(objx,gen,Boolean.TYPE);
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    return Boolean.TYPE;
  }

}

static class InvokeExpr implements Expr{
  public final Expr fexpr;
  public final Object tag;
  public final IPersistentVector args;
  public final int line;
  public final String source;
  public boolean isProtocol = false;
  public boolean isDirect = false;
  public int siteIndex = -1;
  public Class protocolOn;
  public java.lang.reflect.Method onMethod;
  static Keyword onKey = Keyword.intern("on");
  static Keyword methodMapKey = Keyword.intern("method-map");
  static Keyword dynamicKey = Keyword.intern("dynamic");

  public InvokeExpr(String source, int line, Symbol tag, Expr fexpr, IPersistentVector args) throws Exception{
    this.source = source;
    this.fexpr = fexpr;
    this.args = args;
    this.line = line;
    if(fexpr instanceof VarExpr)
      {
      Var fvar = ((VarExpr)fexpr).var;
      Var pvar =  (Var)RT.get(fvar.meta(), protocolKey);
      if(pvar != null && PROTOCOL_CALLSITES.isBound())
        {
        this.isProtocol = true;
        this.siteIndex = registerProtocolCallsite(((VarExpr)fexpr).var);
        Object pon = RT.get(pvar.get(), onKey);
        this.protocolOn = HostExpr.maybeClass(pon,false);
        if(this.protocolOn != null)
          {
          IPersistentMap mmap = (IPersistentMap) RT.get(pvar.get(), methodMapKey);
                    Keyword mmapVal = (Keyword) mmap.valAt(Keyword.intern(fvar.sym));
                    if (mmapVal == null) {
                        throw new IllegalArgumentException(
                              "No method of interface: " + protocolOn.getName() +
                              " found for function: " + fvar.sym + " of protocol: " + pvar.sym +
                              " (The protocol method may have been defined before and removed.)");
                    }
                    String mname = munge(mmapVal.sym.toString());
           List methods = Reflector.getMethods(protocolOn, args.count() - 1, mname, false);
          if(methods.size() != 1)
            throw new IllegalArgumentException(
                "No single method: " + mname + " of interface: " + protocolOn.getName() +
                " found for function: " + fvar.sym + " of protocol: " + pvar.sym);
          this.onMethod = (java.lang.reflect.Method) methods.get(0);
          }
        }
//      else if(pvar == null && VAR_CALLSITES.isBound()
//              && fvar.ns.name.name.startsWith("clojure")
//          && !RT.booleanCast(RT.get(RT.meta(fvar),dynamicKey))
//          )
//        {
//        //todo - more specific criteria for binding these
//        this.isDirect = true;
//        this.siteIndex = registerVarCallsite(((VarExpr) fexpr).var);
//        }
      }
    this.tag = tag != null ? tag : (fexpr instanceof VarExpr ? ((VarExpr) fexpr).tag : null);
  }

  public Object eval() throws Exception{
    try
      {
      IFn fn = (IFn) fexpr.eval();
      PersistentVector argvs = PersistentVector.EMPTY;
      for(int i = 0; i < args.count(); i++)
        argvs = argvs.cons(((Expr) args.nth(i)).eval());
      return fn.applyTo(RT.seq(argvs));
      }
    catch(Throwable e)
      {
      if(!(e instanceof CompilerException))
        throw new CompilerException(source, line, e);
      else
        throw (CompilerException) e;
      }
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    gen.visitLineNumber(line, gen.mark());
    if(isProtocol)
      {
      emitProto(context,objx,gen);
      }
    else if(isDirect)
      {
      Label callLabel = gen.newLabel();

      gen.getStatic(objx.objtype, objx.varCallsiteName(siteIndex), IFN_TYPE);
      gen.dup();
      gen.ifNonNull(callLabel);

      gen.pop();
      fexpr.emit(C.EXPRESSION, objx, gen);
      gen.checkCast(IFN_TYPE);
//      gen.dup();
//      gen.putStatic(objx.objtype, objx.varCallsiteName(siteIndex), IFN_TYPE);

      gen.mark(callLabel);
      emitArgsAndCall(0, context,objx,gen);
      }
    else
      {
      fexpr.emit(C.EXPRESSION, objx, gen);
      gen.checkCast(IFN_TYPE);
      emitArgsAndCall(0, context,objx,gen);
      }
    if(context == C.STATEMENT)
      gen.pop();   
  }

  public void emitProto(C context, ObjExpr objx, GeneratorAdapter gen){
    Label onLabel = gen.newLabel();
    Label callLabel = gen.newLabel();
    Label endLabel = gen.newLabel();

    Var v = ((VarExpr)fexpr).var;

    Expr e = (Expr) args.nth(0);
    e.emit(C.EXPRESSION, objx, gen);
    gen.dup(); //target, target
    gen.invokeStatic(UTIL_TYPE,Method.getMethod("Class classOf(Object)")); //target,class
    gen.loadThis();
    gen.getField(objx.objtype, objx.cachedClassName(siteIndex),CLASS_TYPE); //target,class,cached-class
    gen.visitJumpInsn(IF_ACMPEQ, callLabel); //target
    if(protocolOn != null)
      {
      gen.dup(); //target, target     
      gen.instanceOf(Type.getType(protocolOn));
      gen.ifZCmp(GeneratorAdapter.NE, onLabel);
      }

    gen.mark(callLabel); //target
    gen.dup(); //target, target
    gen.invokeStatic(UTIL_TYPE,Method.getMethod("Class classOf(Object)")); //target,class
    gen.loadThis();
    gen.swap();
    gen.putField(objx.objtype, objx.cachedClassName(siteIndex),CLASS_TYPE); //target
    objx.emitVar(gen, v);
    gen.invokeVirtual(VAR_TYPE, Method.getMethod("Object getRawRoot()")); //target, proto-fn
    gen.swap();
    emitArgsAndCall(1, context,objx,gen);
    gen.goTo(endLabel);

    gen.mark(onLabel); //target
    if(protocolOn != null)
      {
      MethodExpr.emitTypedArgs(objx, gen, onMethod.getParameterTypes(), RT.subvec(args,1,args.count()));
      if(context == C.RETURN)
        {
        ObjMethod method = (ObjMethod) METHOD.deref();
        method.emitClearLocals(gen);
        }
      Method m = new Method(onMethod.getName(), Type.getReturnType(onMethod), Type.getArgumentTypes(onMethod));
      gen.invokeInterface(Type.getType(protocolOn), m);
      HostExpr.emitBoxReturn(objx, gen, onMethod.getReturnType());
      }
    gen.mark(endLabel);
  }

  void emitArgsAndCall(int firstArgToEmit, C context, ObjExpr objx, GeneratorAdapter gen){
    for(int i = firstArgToEmit; i < Math.min(MAX_POSITIONAL_ARITY, args.count()); i++)
      {
      Expr e = (Expr) args.nth(i);
      e.emit(C.EXPRESSION, objx, gen);
      }
    if(args.count() > MAX_POSITIONAL_ARITY)
      {
      PersistentVector restArgs = PersistentVector.EMPTY;
      for(int i = MAX_POSITIONAL_ARITY; i < args.count(); i++)
        {
        restArgs = restArgs.cons(args.nth(i));
        }
      MethodExpr.emitArgsAsArray(restArgs, objx, gen);
      }

    if(context == C.RETURN)
      {
      ObjMethod method = (ObjMethod) METHOD.deref();
      method.emitClearLocals(gen);
      }

    gen.invokeInterface(IFN_TYPE, new Method("invoke", OBJECT_TYPE, ARG_TYPES[Math.min(MAX_POSITIONAL_ARITY + 1,
                                                                                       args.count())]));
  }

  public boolean hasJavaClass() throws Exception{
    return tag != null;
  }

  public Class getJavaClass() throws Exception{
    return HostExpr.tagToClass(tag);
  }

  static public Expr parse(C context, ISeq form) throws Exception{
    if(context != C.EVAL)
      context = C.EXPRESSION;
    Expr fexpr = analyze(context, form.first());
    if(fexpr instanceof VarExpr && ((VarExpr)fexpr).var.equals(INSTANCE))
      {
      if(RT.second(form) instanceof Symbol)
        {
        Class c = HostExpr.maybeClass(RT.second(form),false);
        if(c != null)
          return new InstanceOfExpr(c, analyze(context, RT.third(form)));
        }
      }

    if(fexpr instanceof KeywordExpr && RT.count(form) == 2 && KEYWORD_CALLSITES.isBound())
      {
//      fexpr = new ConstantExpr(new KeywordCallSite(((KeywordExpr)fexpr).k));
      Expr target = analyze(context, RT.second(form));
      return new KeywordInvokeExpr((String) SOURCE.deref(), (Integer) LINE.deref(), tagOf(form),
                                   (KeywordExpr) fexpr, target);
      }
    PersistentVector args = PersistentVector.EMPTY;
    for(ISeq s = RT.seq(form.next()); s != null; s = s.next())
      {
      args = args.cons(analyze(context, s.first()));
      }
//    if(args.count() > MAX_POSITIONAL_ARITY)
//      throw new IllegalArgumentException(
//          String.format("No more than %d args supported", MAX_POSITIONAL_ARITY));

    return new InvokeExpr((String) SOURCE.deref(), (Integer) LINE.deref(), tagOf(form), fexpr, args);
  }
}

static class SourceDebugExtensionAttribute extends Attribute{
  public SourceDebugExtensionAttribute(){
    super("SourceDebugExtension");
  }

  void writeSMAP(ClassWriter cw, String smap){
    ByteVector bv = write(cw, null, -1, -1, -1);
    bv.putUTF8(smap);
  }
}

static public class FnExpr extends ObjExpr{
  final static Type aFnType = Type.getType(AFunction.class);
  final static Type restFnType = Type.getType(RestFn.class);
  //if there is a variadic overload (there can only be one) it is stored here
  FnMethod variadicMethod = null;
  IPersistentCollection methods;
  //  String superName = null;

  public FnExpr(Object tag){
    super(tag);
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    return AFunction.class;
  }

  protected void emitMethods(ClassVisitor cv){
    //override of invoke/doInvoke for each method
    for(ISeq s = RT.seq(methods); s != null; s = s.next())
      {
      ObjMethod method = (ObjMethod) s.first();
      method.emit(this, cv);
      }

    if(isVariadic())
      {
      GeneratorAdapter gen = new GeneratorAdapter(ACC_PUBLIC,
                                                  Method.getMethod("int getRequiredArity()"),
                                                  null,
                                                  null,
                                                  cv);
      gen.visitCode();
      gen.push(variadicMethod.reqParms.count());
      gen.returnValue();
      gen.endMethod();
      }
  }

  static Expr parse(C context, ISeq form, String name) throws Exception{
    ISeq origForm = form;
    FnExpr fn = new FnExpr(tagOf(form));
    fn.src = form;
    ObjMethod enclosingMethod = (ObjMethod) METHOD.deref();
    if(((IMeta) form.first()).meta() != null)
      {
      fn.onceOnly = RT.booleanCast(RT.get(RT.meta(form.first()), Keyword.intern(null, "once")));
//      fn.superName = (String) RT.get(RT.meta(form.first()), Keyword.intern(null, "super-name"));
      }
    //fn.thisName = name;
    String basename = enclosingMethod != null ?
                      (enclosingMethod.objx.name + "$")
                                              : //"clojure.fns." +
                      (munge(currentNS().name.name) + "$");
    if(RT.second(form) instanceof Symbol)
      name = ((Symbol) RT.second(form)).name;
    String simpleName = name != null ?
                        (munge(name).replace(".", "_DOT_")
                        + (enclosingMethod != null ? "__" + RT.nextID() : ""))
                        : ("fn"
                          + "__" + RT.nextID());
    fn.name = basename + simpleName;
    fn.internalName = fn.name.replace('.', '/');
    fn.objtype = Type.getObjectType(fn.internalName);
    try
      {
      Var.pushThreadBindings(
          RT.map(CONSTANTS, PersistentVector.EMPTY,
                 CONSTANT_IDS, new IdentityHashMap(),
                 KEYWORDS, PersistentHashMap.EMPTY,
                 VARS, PersistentHashMap.EMPTY,
                 KEYWORD_CALLSITES, PersistentVector.EMPTY,
                 PROTOCOL_CALLSITES, PersistentVector.EMPTY,
                 VAR_CALLSITES, PersistentVector.EMPTY
          ));

      //arglist might be preceded by symbol naming this fn
      if(RT.second(form) instanceof Symbol)
        {
        fn.thisName = ((Symbol) RT.second(form)).name;
        form = RT.cons(FN, RT.next(RT.next(form)));
        }

      //now (fn [args] body...) or (fn ([args] body...) ([args2] body2...) ...)
      //turn former into latter
      if(RT.second(form) instanceof IPersistentVector)
        form = RT.list(FN, RT.next(form));
      fn.line = (Integer) LINE.deref();
      FnMethod[] methodArray = new FnMethod[MAX_POSITIONAL_ARITY + 1];
      FnMethod variadicMethod = null;
      for(ISeq s = RT.next(form); s != null; s = RT.next(s))
        {
        FnMethod f = FnMethod.parse(fn, (ISeq) RT.first(s));
        if(f.isVariadic())
          {
          if(variadicMethod == null)
            variadicMethod = f;
          else
            throw new Exception("Can't have more than 1 variadic overload");
          }
        else if(methodArray[f.reqParms.count()] == null)
          methodArray[f.reqParms.count()] = f;
        else
          throw new Exception("Can't have 2 overloads with same arity");
        }
      if(variadicMethod != null)
        {
        for(int i = variadicMethod.reqParms.count() + 1; i <= MAX_POSITIONAL_ARITY; i++)
          if(methodArray[i] != null)
            throw new Exception(
                "Can't have fixed arity function with more params than variadic function");
        }

      IPersistentCollection methods = null;
      for(int i = 0; i < methodArray.length; i++)
        if(methodArray[i] != null)
          methods = RT.conj(methods, methodArray[i]);
      if(variadicMethod != null)
        methods = RT.conj(methods, variadicMethod);

      fn.methods = methods;
      fn.variadicMethod = variadicMethod;
      fn.keywords = (IPersistentMap) KEYWORDS.deref();
      fn.vars = (IPersistentMap) VARS.deref();
      fn.constants = (PersistentVector) CONSTANTS.deref();
      fn.keywordCallsites = (IPersistentVector) KEYWORD_CALLSITES.deref();
      fn.protocolCallsites = (IPersistentVector) PROTOCOL_CALLSITES.deref();
      fn.varCallsites = (IPersistentVector) VAR_CALLSITES.deref();

      fn.constantsID = RT.nextID();
//      DynamicClassLoader loader = (DynamicClassLoader) LOADER.get();
//      loader.registerConstants(fn.constantsID, fn.constants.toArray());
      }
    finally
      {
      Var.popThreadBindings();
      }
    fn.compile(fn.isVariadic() ? "clojure/lang/RestFn" : "clojure/lang/AFunction",null,fn.onceOnly);
    fn.getCompiledClass();

    if(origForm instanceof IObj && ((IObj) origForm).meta() != null)
      return new MetaExpr(fn, (MapExpr) MapExpr
          .parse(context == C.EVAL ? context : C.EXPRESSION, ((IObj) origForm).meta()));
    else
      return fn;
  }

  public final ObjMethod variadicMethod(){
    return variadicMethod;
  }

  boolean isVariadic(){
    return variadicMethod != null;
  }

  public final IPersistentCollection methods(){
    return methods;
  }
}

static public class ObjExpr implements Expr{
  static final String CONST_PREFIX = "const__";
  String name;
  //String simpleName;
  String internalName;
  String thisName;
  Type objtype;
  public final Object tag;
  //localbinding->itself
  IPersistentMap closes = PersistentHashMap.EMPTY;
    //localbndingexprs
    IPersistentVector closesExprs = PersistentVector.EMPTY;
  //symbols
  IPersistentSet volatiles = PersistentHashSet.EMPTY;

  //symbol->lb
  IPersistentMap fields = null;

  //Keyword->KeywordExpr
  IPersistentMap keywords = PersistentHashMap.EMPTY;
  IPersistentMap vars = PersistentHashMap.EMPTY;
  Class compiledClass;
  int line;
  PersistentVector constants;
  int constantsID;
  int altCtorDrops = 0;

  IPersistentVector keywordCallsites;
  IPersistentVector protocolCallsites;
  IPersistentVector varCallsites;
  boolean onceOnly = false;

  Object src;

  final static Method voidctor = Method.getMethod("void <init>()");
  protected IPersistentMap classMeta;

  public final String name(){
    return name;
  }

//  public final String simpleName(){
//    return simpleName;
//  }

  public final String internalName(){
    return internalName;
  }

  public final String thisName(){
    return thisName;
  }

  public final Type objtype(){
    return objtype;
  }

  public final IPersistentMap closes(){
    return closes;
  }

  public final IPersistentMap keywords(){
    return keywords;
  }

  public final IPersistentMap vars(){
    return vars;
  }

  public final Class compiledClass(){
    return compiledClass;
  }

  public final int line(){
    return line;
  }

  public final PersistentVector constants(){
    return constants;
  }

  public final int constantsID(){
    return constantsID;
  }

  final static Method kwintern = Method.getMethod("clojure.lang.Keyword intern(String, String)");
  final static Method symcreate = Method.getMethod("clojure.lang.Symbol create(String)");
  final static Method varintern =
      Method.getMethod("clojure.lang.Var intern(clojure.lang.Symbol, clojure.lang.Symbol)");

  final static Type DYNAMIC_CLASSLOADER_TYPE = Type.getType(DynamicClassLoader.class);
  final static Method getClassMethod = Method.getMethod("Class getClass()");
  final static Method getClassLoaderMethod = Method.getMethod("ClassLoader getClassLoader()");
  final static Method getConstantsMethod = Method.getMethod("Object[] getConstants(int)");
  final static Method readStringMethod = Method.getMethod("Object readString(String)");

  final static Type ILOOKUP_SITE_TYPE = Type.getType(ILookupSite.class);
  final static Type ILOOKUP_THUNK_TYPE = Type.getType(ILookupThunk.class);
  final static Type KEYWORD_LOOKUPSITE_TYPE = Type.getType(KeywordLookupSite.class);

  private DynamicClassLoader loader;
  private byte[] bytecode;

  public ObjExpr(Object tag){
    this.tag = tag;
  }

  static String trimGenID(String name){
    int i = name.lastIndexOf("__");
    return i==-1?name:name.substring(0,i);
  }
 


  Type[] ctorTypes(){
    IPersistentVector tv = isDeftype()?PersistentVector.EMPTY:RT.vector(IPERSISTENTMAP_TYPE);
    for(ISeq s = RT.keys(closes); s != null; s = s.next())
      {
      LocalBinding lb = (LocalBinding) s.first();
      if(lb.getPrimitiveType() != null)
        tv = tv.cons(Type.getType(lb.getPrimitiveType()));
      else
        tv = tv.cons(OBJECT_TYPE);
      }
    Type[] ret = new Type[tv.count()];
    for(int i = 0; i < tv.count(); i++)
      ret[i] = (Type) tv.nth(i);
    return ret;
  }

  void compile(String superName, String[] interfaceNames, boolean oneTimeUse) throws Exception{
    //create bytecode for a class
    //with name current_ns.defname[$letname]+
    //anonymous fns get names fn__id
    //derived from AFn/RestFn
    if(keywordCallsites.count() > 0)
      {
      if(interfaceNames == null)
        interfaceNames = new String[]{"clojure/lang/ILookupHost"};
      else
        {
        String[] inames = new String[interfaceNames.length + 1];
        System.arraycopy(interfaceNames,0,inames,0,interfaceNames.length);
        inames[interfaceNames.length] "clojure/lang/ILookupHost";
        interfaceNames = inames;
        }
      }
    ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
//    ClassWriter cw = new ClassWriter(0);
    ClassVisitor cv = cw;
//    ClassVisitor cv = new TraceClassVisitor(new CheckClassAdapter(cw), new PrintWriter(System.out));
    //ClassVisitor cv = new TraceClassVisitor(cw, new PrintWriter(System.out));
    cv.visit(V1_5, ACC_PUBLIC + ACC_SUPER + ACC_FINAL, internalName, null,superName,interfaceNames);
//             superName != null ? superName :
//             (isVariadic() ? "clojure/lang/RestFn" : "clojure/lang/AFunction"), null);
    String source = (String) SOURCE.deref();
    int lineBefore = (Integer) LINE_BEFORE.deref();
    int lineAfter = (Integer) LINE_AFTER.deref() + 1;

    if(source != null && SOURCE_PATH.deref() != null)
      {
      //cv.visitSource(source, null);
      String smap = "SMAP\n" +
                    ((source.lastIndexOf('.') > 0) ?
                     source.substring(0, source.lastIndexOf('.'))
                      :source)
                             //                      : simpleName)
                    + ".java\n" +
                    "Clojure\n" +
                    "*S Clojure\n" +
                    "*F\n" +
                    "+ 1 " + source + "\n" +
                    (String) SOURCE_PATH.deref() + "\n" +
                    "*L\n" +
                    String.format("%d#1,%d:%d\n", lineBefore, lineAfter - lineBefore, lineBefore) +
                    "*E";
      cv.visitSource(source, smap);
      }
    addAnnotation(cv, classMeta);
    //static fields for constants
    for(int i = 0; i < constants.count(); i++)
      {
      cv.visitField(ACC_PUBLIC + ACC_FINAL
                    + ACC_STATIC, constantName(i), constantType(i).getDescriptor(),
                    null, null);
      }

    //static fields for lookup sites
    for(int i = 0; i < keywordCallsites.count(); i++)
      {
      cv.visitField(ACC_FINAL
                    + ACC_STATIC, siteNameStatic(i), KEYWORD_LOOKUPSITE_TYPE.getDescriptor(),
                    null, null);
      cv.visitField(ACC_STATIC, thunkNameStatic(i), ILOOKUP_THUNK_TYPE.getDescriptor(),
                    null, null);
      }

    for(int i=0;i<varCallsites.count();i++)
      {
      cv.visitField(ACC_PRIVATE + ACC_STATIC + ACC_FINAL
          , varCallsiteName(i), IFN_TYPE.getDescriptor(), null, null);
      }

    //static init for constants, keywords and vars
    GeneratorAdapter clinitgen = new GeneratorAdapter(ACC_PUBLIC + ACC_STATIC,
                                                      Method.getMethod("void <clinit> ()"),
                                                      null,
                                                      null,
                                                      cv);
    clinitgen.visitCode();
    clinitgen.visitLineNumber(line, clinitgen.mark());

    if(constants.count() > 0)
      {
      emitConstants(clinitgen);
      }

    if(keywordCallsites.count() > 0)
      emitKeywordCallsites(clinitgen);

    for(int i=0;i<varCallsites.count();i++)
      {
      Label skipLabel = clinitgen.newLabel();
      Label endLabel = clinitgen.newLabel();
      Var var = (Var) varCallsites.nth(i);
      clinitgen.push(var.ns.name.toString());
      clinitgen.push(var.sym.toString());
      clinitgen.invokeStatic(RT_TYPE, Method.getMethod("clojure.lang.Var var(String,String)"));
      clinitgen.dup();
      clinitgen.invokeVirtual(VAR_TYPE,Method.getMethod("boolean hasRoot()"));
      clinitgen.ifZCmp(GeneratorAdapter.EQ,skipLabel);

      clinitgen.invokeVirtual(VAR_TYPE,Method.getMethod("Object getRoot()"));
            clinitgen.dup();
            clinitgen.instanceOf(AFUNCTION_TYPE);
            clinitgen.ifZCmp(GeneratorAdapter.EQ,skipLabel);
      clinitgen.checkCast(IFN_TYPE);
      clinitgen.putStatic(objtype, varCallsiteName(i), IFN_TYPE);
      clinitgen.goTo(endLabel);

      clinitgen.mark(skipLabel);
      clinitgen.pop();

      clinitgen.mark(endLabel);
      }

    clinitgen.returnValue();

    clinitgen.endMethod();
    if(!isDeftype())
      {
      cv.visitField(ACC_FINAL, "__meta", IPERSISTENTMAP_TYPE.getDescriptor(), null, null);
      }
    //instance fields for closed-overs
    for(ISeq s = RT.keys(closes); s != null; s = s.next())
      {
      LocalBinding lb = (LocalBinding) s.first();
      if(isDeftype())
        {
        int access = isVolatile(lb) ? ACC_VOLATILE :
                     isMutable(lb) ? 0 :
                     (ACC_PUBLIC + ACC_FINAL);
        FieldVisitor fv;
        if(lb.getPrimitiveType() != null)
          fv = cv.visitField(access
              , lb.name, Type.getType(lb.getPrimitiveType()).getDescriptor(),
                  null, null);
        else
        //todo - when closed-overs are fields, use more specific types here and in ctor and emitLocal?
          fv = cv.visitField(access
              , lb.name, OBJECT_TYPE.getDescriptor(), null, null);
        addAnnotation(fv, RT.meta(lb.sym));
        }
      else
        {
        //todo - only enable this non-private+writability for letfns where we need it
        if(lb.getPrimitiveType() != null)
          cv.visitField(0 + (isVolatile(lb) ? ACC_VOLATILE : 0)
              , lb.name, Type.getType(lb.getPrimitiveType()).getDescriptor(),
                  null, null);
        else
          cv.visitField(0 //+ (oneTimeUse ? 0 : ACC_FINAL)
              , lb.name, OBJECT_TYPE.getDescriptor(), null, null);
        }
      }

    //instance fields for callsites and thunks
    for(int i=0;i<protocolCallsites.count();i++)
      {
      cv.visitField(ACC_PRIVATE, cachedClassName(i), CLASS_TYPE.getDescriptor(), null, null);
      cv.visitField(ACC_PRIVATE, cachedProtoFnName(i), AFUNCTION_TYPE.getDescriptor(), null, null);
      cv.visitField(ACC_PRIVATE, cachedProtoImplName(i), IFN_TYPE.getDescriptor(), null, null);     
      }

    //ctor that takes closed-overs and inits base + fields
    Method m = new Method("<init>", Type.VOID_TYPE, ctorTypes());
    GeneratorAdapter ctorgen = new GeneratorAdapter(ACC_PUBLIC,
                                                    m,
                                                    null,
                                                    null,
                                                    cv);
    Label start = ctorgen.newLabel();
    Label end = ctorgen.newLabel();
    ctorgen.visitCode();
    ctorgen.visitLineNumber(line, ctorgen.mark());
    ctorgen.visitLabel(start);
    ctorgen.loadThis();
//    if(superName != null)
      ctorgen.invokeConstructor(Type.getObjectType(superName), voidctor);
//    else if(isVariadic()) //RestFn ctor takes reqArity arg
//      {
//      ctorgen.push(variadicMethod.reqParms.count());
//      ctorgen.invokeConstructor(restFnType, restfnctor);
//      }
//    else
//      ctorgen.invokeConstructor(aFnType, voidctor);
    if(!isDeftype())
      {
      ctorgen.loadThis();
      ctorgen.visitVarInsn(IPERSISTENTMAP_TYPE.getOpcode(Opcodes.ILOAD), 1);
      ctorgen.putField(objtype, "__meta", IPERSISTENTMAP_TYPE);
      }

    int a = isDeftype()?1:2;
    for(ISeq s = RT.keys(closes); s != null; s = s.next(), ++a)
      {
      LocalBinding lb = (LocalBinding) s.first();
      ctorgen.loadThis();
      Class primc = lb.getPrimitiveType();
      if(primc != null)
        {
        ctorgen.visitVarInsn(Type.getType(primc).getOpcode(Opcodes.ILOAD), a);
        ctorgen.putField(objtype, lb.name, Type.getType(primc));
        if(primc == Long.TYPE || primc == Double.TYPE)
          ++a;
        }
      else
        {
        ctorgen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ILOAD), a);
        ctorgen.putField(objtype, lb.name, OBJECT_TYPE);
        }
            closesExprs = closesExprs.cons(new LocalBindingExpr(lb, null));
      }


    ctorgen.visitLabel(end);

    ctorgen.returnValue();

    ctorgen.endMethod();

    if(altCtorDrops > 0)
      {
          //ctor that takes closed-overs and inits base + fields
      Type[] ctorTypes = ctorTypes();
      Type[] altCtorTypes = new Type[ctorTypes.length-altCtorDrops];
      for(int i=0;i<altCtorTypes.length;i++)
        altCtorTypes[i] = ctorTypes[i];
      Method alt = new Method("<init>", Type.VOID_TYPE, altCtorTypes);
      ctorgen = new GeneratorAdapter(ACC_PUBLIC,
                              alt,
                              null,
                              null,
                              cv);
      ctorgen.visitCode();
      ctorgen.loadThis();
      ctorgen.loadArgs();
      for(int i=0;i<altCtorDrops;i++)
        ctorgen.visitInsn(Opcodes.ACONST_NULL);

      ctorgen.invokeConstructor(objtype, new Method("<init>", Type.VOID_TYPE, ctorTypes));

      ctorgen.returnValue();
      ctorgen.endMethod();
      }

    if(!isDeftype())
      {
          //ctor that takes closed-overs but not meta
      Type[] ctorTypes = ctorTypes();
      Type[] noMetaCtorTypes = new Type[ctorTypes.length-1];
      for(int i=1;i<ctorTypes.length;i++)
        noMetaCtorTypes[i-1] = ctorTypes[i];
      Method alt = new Method("<init>", Type.VOID_TYPE, noMetaCtorTypes);
      ctorgen = new GeneratorAdapter(ACC_PUBLIC,
                              alt,
                              null,
                              null,
                              cv);
      ctorgen.visitCode();
      ctorgen.loadThis();
      ctorgen.visitInsn(Opcodes.ACONST_NULL)//null meta
      ctorgen.loadArgs();
      ctorgen.invokeConstructor(objtype, new Method("<init>", Type.VOID_TYPE, ctorTypes));

      ctorgen.returnValue();
      ctorgen.endMethod();

      //meta()
      Method meth = Method.getMethod("clojure.lang.IPersistentMap meta()");

      GeneratorAdapter gen = new GeneratorAdapter(ACC_PUBLIC,
                        meth,
                        null,
                        null,
                        cv);
      gen.visitCode();
      gen.loadThis();
      gen.getField(objtype,"__meta",IPERSISTENTMAP_TYPE);

      gen.returnValue();
      gen.endMethod();

      //withMeta()
      meth = Method.getMethod("clojure.lang.IObj withMeta(clojure.lang.IPersistentMap)");

      gen = new GeneratorAdapter(ACC_PUBLIC,
                        meth,
                        null,
                        null,
                        cv);
      gen.visitCode();
      gen.newInstance(objtype);
      gen.dup();
      gen.loadArg(0);

      for(ISeq s = RT.keys(closes); s != null; s = s.next(), ++a)
        {
        LocalBinding lb = (LocalBinding) s.first();
        gen.loadThis();
        Class primc = lb.getPrimitiveType();
        if(primc != null)
          {
          gen.getField(objtype, lb.name, Type.getType(primc));
          }
        else
          {
          gen.getField(objtype, lb.name, OBJECT_TYPE);
          }
        }

      gen.invokeConstructor(objtype, new Method("<init>", Type.VOID_TYPE, ctorTypes));
      gen.returnValue();
      gen.endMethod();
      }
   
    emitMethods(cv);

    if(keywordCallsites.count() > 0)
      {
      Method meth = Method.getMethod("void swapThunk(int,clojure.lang.ILookupThunk)");

      GeneratorAdapter gen = new GeneratorAdapter(ACC_PUBLIC,
                        meth,
                        null,
                        null,
                        cv);
      gen.visitCode();
      Label endLabel = gen.newLabel();

      Label[] labels = new Label[keywordCallsites.count()];
      for(int i = 0; i < keywordCallsites.count();i++)
        {
        labels[i] = gen.newLabel();
        }
      gen.loadArg(0);
      gen.visitTableSwitchInsn(0,keywordCallsites.count()-1,endLabel,labels);

      for(int i = 0; i < keywordCallsites.count();i++)
        {
        gen.mark(labels[i]);
//        gen.loadThis();
        gen.loadArg(1);
        gen.putStatic(objtype, thunkNameStatic(i),ILOOKUP_THUNK_TYPE);
        gen.goTo(endLabel);
        }

      gen.mark(endLabel);

      gen.returnValue();
      gen.endMethod();
      }
   
    //end of class
    cv.visitEnd();

    bytecode = cw.toByteArray();
    if(RT.booleanCast(COMPILE_FILES.deref()))
      writeClassFile(internalName, bytecode);
//    else
//      getCompiledClass();
  }

  private void emitKeywordCallsites(GeneratorAdapter clinitgen){
    for(int i=0;i<keywordCallsites.count();i++)
      {
      Keyword k = (Keyword) keywordCallsites.nth(i);
      clinitgen.newInstance(KEYWORD_LOOKUPSITE_TYPE);
      clinitgen.dup();
      clinitgen.push(i);
      emitValue(k,clinitgen);
      clinitgen.invokeConstructor(KEYWORD_LOOKUPSITE_TYPE,
                                  Method.getMethod("void <init>(int,clojure.lang.Keyword)"));
      clinitgen.dup();
      clinitgen.putStatic(objtype, siteNameStatic(i), KEYWORD_LOOKUPSITE_TYPE);
      clinitgen.putStatic(objtype, thunkNameStatic(i), ILOOKUP_THUNK_TYPE);
      }
  }

  protected void emitMethods(ClassVisitor gen){
  }

  void emitListAsObjectArray(Object value, GeneratorAdapter gen){
    gen.push(((List) value).size());
    gen.newArray(OBJECT_TYPE);
    int i = 0;
    for(Iterator it = ((List) value).iterator(); it.hasNext(); i++)
      {
      gen.dup();
      gen.push(i);
      emitValue(it.next(), gen);
      gen.arrayStore(OBJECT_TYPE);
      }
  }

  void emitValue(Object value, GeneratorAdapter gen){
    boolean partial = true;
    //System.out.println(value.getClass().toString());

    if(value instanceof String)
      {
      gen.push((String) value);
      }
    else if(value instanceof Integer)
      {
      gen.push(((Integer) value).intValue());
      gen.invokeStatic(Type.getType(Integer.class), Method.getMethod("Integer valueOf(int)"));
      }
    else if(value instanceof Double)
        {
        gen.push(((Double) value).doubleValue());
        gen.invokeStatic(Type.getType(Double.class), Method.getMethod("Double valueOf(double)"));
        }
      else if(value instanceof Character)
          {
          gen.push(((Character) value).charValue());
          gen.invokeStatic(Type.getType(Character.class), Method.getMethod("Character valueOf(char)"));
          }
        else if(value instanceof Class)
            {
                                                Class cc = (Class)value;
                                                if(cc.isPrimitive())
                                                        {
                                                        Type bt;
                                                        if ( cc == boolean.class ) bt = Type.getType(Boolean.class);
                                                        else if ( cc == byte.class ) bt = Type.getType(Byte.class);
                                                        else if ( cc == char.class ) bt = Type.getType(Character.class);
                                                        else if ( cc == double.class ) bt = Type.getType(Double.class);
                                                        else if ( cc == float.class ) bt = Type.getType(Float.class);
                                                        else if ( cc == int.class ) bt = Type.getType(Integer.class);
                                                        else if ( cc == long.class ) bt = Type.getType(Long.class);
                                                        else if ( cc == short.class ) bt = Type.getType(Short.class);
                                                        else throw new RuntimeException(
                                                                "Can't embed unknown primitive in code: " + value);
                                                        gen.getStatic( bt, "TYPE", Type.getType(Class.class) );
                                                        }
                                                else
                                                        {
                                                        gen.push(destubClassName(cc.getName()));
                                                        gen.invokeStatic(Type.getType(Class.class), Method.getMethod("Class forName(String)"));
                                                        }
            }
          else if(value instanceof Symbol)
              {
              gen.push(((Symbol) value).ns);
              gen.push(((Symbol) value).name);
              gen.invokeStatic(Type.getType(Symbol.class),
                               Method.getMethod("clojure.lang.Symbol create(String,String)"));
              }
            else if(value instanceof Keyword)
                {
                emitValue(((Keyword) value).sym, gen);
                gen.invokeStatic(Type.getType(Keyword.class),
                                 Method.getMethod("clojure.lang.Keyword intern(clojure.lang.Symbol)"));
                }
//            else if(value instanceof KeywordCallSite)
//                {
//                emitValue(((KeywordCallSite) value).k.sym, gen);
//                gen.invokeStatic(Type.getType(KeywordCallSite.class),
//                                 Method.getMethod("clojure.lang.KeywordCallSite create(clojure.lang.Symbol)"));
//                }
              else if(value instanceof Var)
                  {
                  Var var = (Var) value;
                  gen.push(var.ns.name.toString());
                  gen.push(var.sym.toString());
                  gen.invokeStatic(RT_TYPE, Method.getMethod("clojure.lang.Var var(String,String)"));
                  }
                else if(value instanceof IPersistentMap)
                    {
                    List entries = new ArrayList();
                    for(Map.Entry entry : (Set<Map.Entry>) ((Map) value).entrySet())
                      {
                      entries.add(entry.getKey());
                      entries.add(entry.getValue());
                      }
                    emitListAsObjectArray(entries, gen);
                    gen.invokeStatic(RT_TYPE,
                                     Method.getMethod("clojure.lang.IPersistentMap map(Object[])"));
                    }
                  else if(value instanceof IPersistentVector)
                      {
                      emitListAsObjectArray(value, gen);
                      gen.invokeStatic(RT_TYPE, Method.getMethod(
                          "clojure.lang.IPersistentVector vector(Object[])"));
                      }
                    else if(value instanceof ISeq || value instanceof IPersistentList)
                        {
                        emitListAsObjectArray(value, gen);
                        gen.invokeStatic(Type.getType(java.util.Arrays.class),
                                         Method.getMethod("java.util.List asList(Object[])"));
                        gen.invokeStatic(Type.getType(PersistentList.class),
                                         Method.getMethod(
                                             "clojure.lang.IPersistentList create(java.util.List)"));
                        }
                      else
                        {
                        String cs = null;
                        try
                          {
                          cs = RT.printString(value);
                          //System.out.println("WARNING SLOW CODE: " + value.getClass() + " -> " + cs);
                          }
                        catch(Exception e)
                          {
                          throw new RuntimeException(
                              "Can't embed object in code, maybe print-dup not defined: " +
                              value);
                          }
                        if(cs.length() == 0)
                          throw new RuntimeException(
                              "Can't embed unreadable object in code: " + value);

                        if(cs.startsWith("#<"))
                          throw new RuntimeException(
                              "Can't embed unreadable object in code: " + cs);

                        gen.push(cs);
                        gen.invokeStatic(RT_TYPE, readStringMethod);
                        partial = false;
                        }

    if(partial)
      {
      if(value instanceof IObj && RT.count(((IObj) value).meta()) > 0)
        {
        gen.checkCast(IOBJ_TYPE);
        emitValue(((IObj) value).meta(), gen);
        gen.checkCast(IPERSISTENTMAP_TYPE);
        gen.invokeInterface(IOBJ_TYPE,
                            Method.getMethod("clojure.lang.IObj withMeta(clojure.lang.IPersistentMap)"));
        }
      }
  }


  void emitConstants(GeneratorAdapter clinitgen){
    try
      {
      Var.pushThreadBindings(RT.map(RT.PRINT_DUP, RT.T));

      for(int i = 0; i < constants.count(); i++)
        {
        emitValue(constants.nth(i), clinitgen);
        clinitgen.checkCast(constantType(i));
        clinitgen.putStatic(objtype, constantName(i), constantType(i));
        }
      }
    finally
      {
      Var.popThreadBindings();
      }
  }

  boolean isMutable(LocalBinding lb){
    return isVolatile(lb) ||
           RT.booleanCast(RT.contains(fields, lb.sym)) &&
           RT.booleanCast(RT.get(lb.sym.meta(), Keyword.intern("unsynchronized-mutable")));
  }

  boolean isVolatile(LocalBinding lb){
    return RT.booleanCast(RT.contains(fields, lb.sym)) &&
           RT.booleanCast(RT.get(lb.sym.meta(), Keyword.intern("volatile-mutable")));
  }

  boolean isDeftype(){
    return fields != null;
  }

  void emitClearCloses(GeneratorAdapter gen){
//    int a = 1;
//    for(ISeq s = RT.keys(closes); s != null; s = s.next(), ++a)
//      {
//      LocalBinding lb = (LocalBinding) s.first();
//      Class primc = lb.getPrimitiveType();
//      if(primc == null)
//        {
//        gen.loadThis();
//        gen.visitInsn(Opcodes.ACONST_NULL);
//        gen.putField(objtype, lb.name, OBJECT_TYPE);
//        }
//      }
  }

  synchronized Class getCompiledClass(){
    if(compiledClass == null)
      try
        {
//        if(RT.booleanCast(COMPILE_FILES.deref()))
//          compiledClass = RT.classForName(name);//loader.defineClass(name, bytecode);
//        else
          {
          loader = (DynamicClassLoader) LOADER.deref();
          compiledClass = loader.defineClass(name, bytecode, src);
          }
        }
      catch(Exception e)
        {
        throw new RuntimeException(e);
        }
    return compiledClass;
  }

  public Object eval() throws Exception{
    if(isDeftype())
      return null;
    return getCompiledClass().newInstance();
  }

  public void emitLetFnInits(GeneratorAdapter gen, ObjExpr objx, IPersistentSet letFnLocals){
    //objx arg is enclosing objx, not this
    gen.checkCast(objtype);

    for(ISeq s = RT.keys(closes); s != null; s = s.next())
      {
      LocalBinding lb = (LocalBinding) s.first();
      if(letFnLocals.contains(lb))
        {
        Class primc = lb.getPrimitiveType();
        gen.dup();
        if(primc != null)
          {
          objx.emitUnboxedLocal(gen, lb);
          gen.putField(objtype, lb.name, Type.getType(primc));
          }
        else
          {
          objx.emitLocal(gen, lb, false);
          gen.putField(objtype, lb.name, OBJECT_TYPE);
          }
        }
      }
    gen.pop();

  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    //emitting a Fn means constructing an instance, feeding closed-overs from enclosing scope, if any
    //objx arg is enclosing objx, not this
//    getCompiledClass();
    if(isDeftype())
      {
      gen.visitInsn(Opcodes.ACONST_NULL);
      }
    else
      {
      gen.newInstance(objtype);
      gen.dup();
      gen.visitInsn(Opcodes.ACONST_NULL);       
      for(ISeq s = RT.seq(closesExprs); s != null; s = s.next())
        {
                LocalBindingExpr lbe = (LocalBindingExpr) s.first();
        LocalBinding lb = lbe.b;
        if(lb.getPrimitiveType() != null)
          objx.emitUnboxedLocal(gen, lb);
        else
          objx.emitLocal(gen, lb, lbe.shouldClear);
        }
      gen.invokeConstructor(objtype, new Method("<init>", Type.VOID_TYPE, ctorTypes()));
      }
    if(context == C.STATEMENT)
      gen.pop();
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    return (compiledClass != null) ? compiledClass
      : (tag != null) ? HostExpr.tagToClass(tag)
      : IFn.class;
  }

  public void emitAssignLocal(GeneratorAdapter gen, LocalBinding lb,Expr val){
    if(!isMutable(lb))
      throw new IllegalArgumentException("Cannot assign to non-mutable: " + lb.name);
    Class primc = lb.getPrimitiveType();
    gen.loadThis();
    if(primc != null)
      {
      if(!(val instanceof MaybePrimitiveExpr && ((MaybePrimitiveExpr) val).canEmitPrimitive()))
        throw new IllegalArgumentException("Must assign primitive to primitive mutable: " + lb.name);
      MaybePrimitiveExpr me = (MaybePrimitiveExpr) val;
      me.emitUnboxed(C.EXPRESSION, this, gen);
      gen.putField(objtype, lb.name, Type.getType(primc));
      }
    else
      {
      val.emit(C.EXPRESSION, this, gen);
      gen.putField(objtype, lb.name, OBJECT_TYPE);
      }
  }

  private void emitLocal(GeneratorAdapter gen, LocalBinding lb, boolean clear){
    if(closes.containsKey(lb))
      {
      Class primc = lb.getPrimitiveType();
      gen.loadThis();
      if(primc != null)
        {
        gen.getField(objtype, lb.name, Type.getType(primc));
        HostExpr.emitBoxReturn(this, gen, primc);
        }
      else
        {
        gen.getField(objtype, lb.name, OBJECT_TYPE);
        if(onceOnly && clear && lb.canBeCleared)
          {
          gen.loadThis();
          gen.visitInsn(Opcodes.ACONST_NULL);
          gen.putField(objtype, lb.name, OBJECT_TYPE);
          }
        }
      }
    else
      {
      Class primc = lb.getPrimitiveType();
//            String rep = lb.sym.name + " " + lb.toString().substring(lb.toString().lastIndexOf('@'));
      if(lb.isArg)
        {
        gen.loadArg(lb.idx-1);
        if(primc != null)
          HostExpr.emitBoxReturn(this, gen, primc);
                else
                    {
                    if(clear && lb.canBeCleared)
                        {
//                        System.out.println("clear: " + rep);
                        gen.visitInsn(Opcodes.ACONST_NULL);
                        gen.storeArg(lb.idx - 1);
                        }
                    else
                        {
//                        System.out.println("use: " + rep);
                        }
                    }    
        }
      else
        {
        if(primc != null)
          {
          gen.visitVarInsn(Type.getType(primc).getOpcode(Opcodes.ILOAD), lb.idx);
          HostExpr.emitBoxReturn(this, gen, primc);
          }
        else
                    {
          gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ILOAD), lb.idx);
                    if(clear && lb.canBeCleared)
                        {
//                        System.out.println("clear: " + rep);
                        gen.visitInsn(Opcodes.ACONST_NULL);
                        gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), lb.idx);
                        }
                    else
                        {
//                        System.out.println("use: " + rep);
                        }
                    }
        }
      }
  }

  private void emitUnboxedLocal(GeneratorAdapter gen, LocalBinding lb){
    Class primc = lb.getPrimitiveType();
    if(closes.containsKey(lb))
      {
      gen.loadThis();
      gen.getField(objtype, lb.name, Type.getType(primc));
      }
    else if(lb.isArg)
      gen.loadArg(lb.idx-1);
    else
      gen.visitVarInsn(Type.getType(primc).getOpcode(Opcodes.ILOAD), lb.idx);
  }

  public void emitVar(GeneratorAdapter gen, Var var){
    Integer i = (Integer) vars.valAt(var);
    emitConstant(gen, i);
    //gen.getStatic(fntype, munge(var.sym.toString()), VAR_TYPE);
  }

  public void emitKeyword(GeneratorAdapter gen, Keyword k){
    Integer i = (Integer) keywords.valAt(k);
    emitConstant(gen, i);
//    gen.getStatic(fntype, munge(k.sym.toString()), KEYWORD_TYPE);
  }

  public void emitConstant(GeneratorAdapter gen, int id){
    gen.getStatic(objtype, constantName(id), constantType(id));
  }


  String constantName(int id){
    return CONST_PREFIX + id;
  }

  String siteName(int n){
    return "__site__" + n;
  }

  String siteNameStatic(int n){
    return siteName(n) + "__";
  }

  String thunkName(int n){
    return "__thunk__" + n;
  }

  String cachedClassName(int n){
    return "__cached_class__" + n;
  }

  String cachedProtoFnName(int n){
    return "__cached_proto_fn__" + n;
  }

  String cachedProtoImplName(int n){
    return "__cached_proto_impl__" + n;
  }

  String varCallsiteName(int n){
    return "__var__callsite__" + n;
  }

  String thunkNameStatic(int n){
    return thunkName(n) + "__";
  }

  Type constantType(int id){
    Object o = constants.nth(id);
    Class c = o.getClass();
    if(Modifier.isPublic(c.getModifiers()))
      {
      //can't emit derived fn types due to visibility
      if(LazySeq.class.isAssignableFrom(c))
        return Type.getType(ISeq.class);
      else if(c == Keyword.class)
        return Type.getType(Keyword.class);
//      else if(c == KeywordCallSite.class)
//        return Type.getType(KeywordCallSite.class);
      else if(RestFn.class.isAssignableFrom(c))
        return Type.getType(RestFn.class);
      else if(AFn.class.isAssignableFrom(c))
          return Type.getType(AFn.class);
        else if(c == Var.class)
            return Type.getType(Var.class);
          else if(c == String.class)
              return Type.getType(String.class);

//      return Type.getType(c);
      }
    return OBJECT_TYPE;
  }

}

enum PATHTYPE {
    PATH, BRANCH;
}

static class PathNode{
    final PATHTYPE type;
    final PathNode parent;

    PathNode(PATHTYPE type, PathNode parent) {
        this.type = type;
        this.parent = parent;
    }
}

static PathNode clearPathRoot(){
    return (PathNode) CLEAR_ROOT.get();
}
   
enum PSTATE{
  REQ, REST, DONE
}

public static class FnMethod extends ObjMethod{
  //localbinding->localbinding
  PersistentVector reqParms = PersistentVector.EMPTY;
  LocalBinding restParm = null;

  public FnMethod(ObjExpr objx, ObjMethod parent){
    super(objx, parent);
  }

  static FnMethod parse(ObjExpr objx, ISeq form) throws Exception{
    //([args] body...)
    IPersistentVector parms = (IPersistentVector) RT.first(form);
    ISeq body = RT.next(form);
    try
      {
      FnMethod method = new FnMethod(objx, (ObjMethod) METHOD.deref());
      method.line = (Integer) LINE.deref();
      //register as the current method and set up a new env frame
            PathNode pnode =  (PathNode) CLEAR_PATH.get();
      if(pnode == null)
        pnode = new PathNode(PATHTYPE.PATH,null);
      Var.pushThreadBindings(
          RT.map(
              METHOD, method,
              LOCAL_ENV, LOCAL_ENV.deref(),
              LOOP_LOCALS, null,
              NEXT_LOCAL_NUM, 0
                            ,CLEAR_PATH, pnode
                            ,CLEAR_ROOT, pnode
                            ,CLEAR_SITES, PersistentHashMap.EMPTY
                        ));

      //register 'this' as local 0
      //registerLocal(THISFN, null, null);
      if(objx.thisName != null)
        registerLocal(Symbol.intern(objx.thisName), null, null,false);
      else
        getAndIncLocalNum();
      PSTATE state = PSTATE.REQ;
      PersistentVector argLocals = PersistentVector.EMPTY;
      for(int i = 0; i < parms.count(); i++)
        {
        if(!(parms.nth(i) instanceof Symbol))
          throw new IllegalArgumentException("fn params must be Symbols");
        Symbol p = (Symbol) parms.nth(i);
        if(p.getNamespace() != null)
          throw new Exception("Can't use qualified name as parameter: " + p);
        if(p.equals(_AMP_))
          {
          if(state == PSTATE.REQ)
            state = PSTATE.REST;
          else
            throw new Exception("Invalid parameter list");
          }

        else
          {
          LocalBinding lb = registerLocal(p, state == PSTATE.REST ? ISEQ : tagOf(p), null,true);
          argLocals = argLocals.cons(lb);
          switch(state)
            {
            case REQ:
              method.reqParms = method.reqParms.cons(lb);
              break;
            case REST:
              method.restParm = lb;
              state = PSTATE.DONE;
              break;

            default:
              throw new Exception("Unexpected parameter");
            }
          }
        }
      if(method.reqParms.count() > MAX_POSITIONAL_ARITY)
        throw new Exception("Can't specify more than " + MAX_POSITIONAL_ARITY + " params");
      LOOP_LOCALS.set(argLocals);
      method.argLocals = argLocals;
      method.body = (new BodyExpr.Parser()).parse(C.RETURN, body);
      return method;
      }
    finally
      {
      Var.popThreadBindings();
      }
  }

  public final PersistentVector reqParms(){
    return reqParms;
  }

  public final LocalBinding restParm(){
    return restParm;
  }

  boolean isVariadic(){
    return restParm != null;
  }

  int numParams(){
    return reqParms.count() + (isVariadic() ? 1 : 0);
  }

  String getMethodName(){
    return isVariadic()?"doInvoke":"invoke";
  }

  Type getReturnType(){
    return OBJECT_TYPE;
  }

  Type[] getArgTypes(){
    if(isVariadic() && reqParms.count() == MAX_POSITIONAL_ARITY)
      {
      Type[] ret = new Type[MAX_POSITIONAL_ARITY + 1];
      for(int i = 0;i<MAX_POSITIONAL_ARITY + 1;i++)
        ret[i] = OBJECT_TYPE;
      return ret;
      }
    return  ARG_TYPES[numParams()];
  }

  void emitClearLocals(GeneratorAdapter gen){
//    for(int i = 1; i < numParams() + 1; i++)
//      {
//      if(!localsUsedInCatchFinally.contains(i))
//        {
//        gen.visitInsn(Opcodes.ACONST_NULL);
//        gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), i);
//        }
//      }
//    for(int i = numParams() + 1; i < maxLocal + 1; i++)
//      {
//      if(!localsUsedInCatchFinally.contains(i))
//        {
//        LocalBinding b = (LocalBinding) RT.get(indexlocals, i);
//        if(b == null || maybePrimitiveType(b.init) == null)
//          {
//          gen.visitInsn(Opcodes.ACONST_NULL);
//          gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), i);
//          }
//        }
//      }
//    if(((FnExpr)objx).onceOnly)
//      {
//      objx.emitClearCloses(gen);
//      }
  }
}

abstract public static class ObjMethod{
  //when closures are defined inside other closures,
  //the closed over locals need to be propagated to the enclosing objx
  public final ObjMethod parent;
  //localbinding->localbinding
  IPersistentMap locals = null;
  //num->localbinding
  IPersistentMap indexlocals = null;
  Expr body = null;
  ObjExpr objx;
  PersistentVector argLocals;
  int maxLocal = 0;
  int line;
  PersistentHashSet localsUsedInCatchFinally = PersistentHashSet.EMPTY;
  protected IPersistentMap methodMeta;

  public final IPersistentMap locals(){
    return locals;
  }

  public final Expr body(){
    return body;
  }

  public final ObjExpr objx(){
    return objx;
  }

  public final PersistentVector argLocals(){
    return argLocals;
  }

  public final int maxLocal(){
    return maxLocal;
  }

  public final int line(){
    return line;
  }

  public ObjMethod(ObjExpr objx, ObjMethod parent){
    this.parent = parent;
    this.objx = objx;
  }

  abstract int numParams();
  abstract String getMethodName();
  abstract Type getReturnType();
  abstract Type[] getArgTypes();

  public void emit(ObjExpr fn, ClassVisitor cv){
    Method m = new Method(getMethodName(), getReturnType(), getArgTypes());

    GeneratorAdapter gen = new GeneratorAdapter(ACC_PUBLIC,
                                                m,
                                                null,
                                                //todo don't hardwire this
                                                EXCEPTION_TYPES,
                                                cv);
    gen.visitCode();
    Label loopLabel = gen.mark();
    gen.visitLineNumber(line, loopLabel);
    try
      {
      Var.pushThreadBindings(RT.map(LOOP_LABEL, loopLabel, METHOD, this));
      body.emit(C.RETURN, fn, gen);
      Label end = gen.mark();
      gen.visitLocalVariable("this", "Ljava/lang/Object;", null, loopLabel, end, 0);
      for(ISeq lbs = argLocals.seq(); lbs != null; lbs = lbs.next())
        {
        LocalBinding lb = (LocalBinding) lbs.first();
        gen.visitLocalVariable(lb.name, "Ljava/lang/Object;", null, loopLabel, end, lb.idx);
        }
      }
    finally
      {
      Var.popThreadBindings();
      }

    gen.returnValue();
    //gen.visitMaxs(1, 1);
    gen.endMethod();
  }

    void emitClearLocals(GeneratorAdapter gen){
    }
   
  void emitClearLocalsOld(GeneratorAdapter gen){
    for(int i=0;i<argLocals.count();i++)
      {
      LocalBinding lb = (LocalBinding) argLocals.nth(i);
      if(!localsUsedInCatchFinally.contains(lb.idx) && lb.getPrimitiveType() == null)
        {
        gen.visitInsn(Opcodes.ACONST_NULL);
        gen.storeArg(lb.idx - 1);       
        }

      }
//    for(int i = 1; i < numParams() + 1; i++)
//      {
//      if(!localsUsedInCatchFinally.contains(i))
//        {
//        gen.visitInsn(Opcodes.ACONST_NULL);
//        gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), i);
//        }
//      }
    for(int i = numParams() + 1; i < maxLocal + 1; i++)
      {
      if(!localsUsedInCatchFinally.contains(i))
        {
        LocalBinding b = (LocalBinding) RT.get(indexlocals, i);
        if(b == null || maybePrimitiveType(b.init) == null)
          {
          gen.visitInsn(Opcodes.ACONST_NULL);
          gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), i);
          }
        }
      }
  }
}

public static class LocalBinding{
  public final Symbol sym;
  public final Symbol tag;
  public Expr init;
  public final int idx;
  public final String name;
  public final boolean isArg;
    public final PathNode clearPathRoot;
  public boolean canBeCleared = true;

    public LocalBinding(int num, Symbol sym, Symbol tag, Expr init, boolean isArg,PathNode clearPathRoot)
                throws Exception{
    if(maybePrimitiveType(init) != null && tag != null)
      throw new UnsupportedOperationException("Can't type hint a local with a primitive initializer");
    this.idx = num;
    this.sym = sym;
    this.tag = tag;
    this.init = init;
    this.isArg = isArg;
        this.clearPathRoot = clearPathRoot;
    name = munge(sym.name);
  }

  public boolean hasJavaClass() throws Exception{
    if(init != null && init.hasJavaClass()
       && Util.isPrimitive(init.getJavaClass())
       && !(init instanceof MaybePrimitiveExpr))
      return false;
    return tag != null
           || (init != null && init.hasJavaClass());
  }

  public Class getJavaClass() throws Exception{
    return tag != null ? HostExpr.tagToClass(tag)
                       : init.getJavaClass();
  }

  public Class getPrimitiveType(){
    return maybePrimitiveType(init);
  }
}

public static class LocalBindingExpr implements Expr, MaybePrimitiveExpr, AssignableExpr{
  public final LocalBinding b;
  public final Symbol tag;

    public final PathNode clearPath;
    public final PathNode clearRoot;
    public boolean shouldClear = false;


  public LocalBindingExpr(LocalBinding b, Symbol tag)
            throws Exception{
    if(b.getPrimitiveType() != null && tag != null)
      throw new UnsupportedOperationException("Can't type hint a primitive local");
    this.b = b;
    this.tag = tag;

        this.clearPath = (PathNode)CLEAR_PATH.get();
        this.clearRoot = (PathNode)CLEAR_ROOT.get();
        IPersistentCollection sites = (IPersistentCollection) RT.get(CLEAR_SITES.get(),b);

        if(b.idx > 0)
            {
//            Object dummy;

            if(sites != null)
                {
                for(ISeq s = sites.seq();s!=null;s = s.next())
                    {
                    LocalBindingExpr o = (LocalBindingExpr) s.first();
                    PathNode common = commonPath(clearPath,o.clearPath);
                    if(common != null && common.type == PATHTYPE.PATH)
                        o.shouldClear = false;
//                    else
//                        dummy = null;
                    }
                }

            if(clearRoot == b.clearPathRoot)
                {
                this.shouldClear = true;
                sites = RT.conj(sites,this);
                CLEAR_SITES.set(RT.assoc(CLEAR_SITES.get(), b, sites));
                }
//            else
//                dummy = null;
            }
       }

  public Object eval() throws Exception{
    throw new UnsupportedOperationException("Can't eval locals");
  }

  public boolean canEmitPrimitive(){
    return b.getPrimitiveType() != null;
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    objx.emitUnboxedLocal(gen, b);
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    if(context != C.STATEMENT)
      objx.emitLocal(gen, b, shouldClear);
  }

  public Object evalAssign(Expr val) throws Exception{
    throw new UnsupportedOperationException("Can't eval locals");
  }

  public void emitAssign(C context, ObjExpr objx, GeneratorAdapter gen, Expr val){
    objx.emitAssignLocal(gen, b,val);
    if(context != C.STATEMENT)
      objx.emitLocal(gen, b, false);
  }

  public boolean hasJavaClass() throws Exception{
    return tag != null || b.hasJavaClass();
  }

  public Class getJavaClass() throws Exception{
    if(tag != null)
      return HostExpr.tagToClass(tag);
    return b.getJavaClass();
  }


}

public static class BodyExpr implements Expr, MaybePrimitiveExpr{
  PersistentVector exprs;

  public final PersistentVector exprs(){
    return exprs;
  }

  public BodyExpr(PersistentVector exprs){
    this.exprs = exprs;
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object frms) throws Exception{
      ISeq forms = (ISeq) frms;
      if(Util.equals(RT.first(forms), DO))
        forms = RT.next(forms);
      PersistentVector exprs = PersistentVector.EMPTY;
      for(; forms != null; forms = forms.next())
        {
        Expr e = (context != C.EVAL &&
                  (context == C.STATEMENT || forms.next() != null)) ?
                 analyze(C.STATEMENT, forms.first())
                                                                    :
                 analyze(context, forms.first());
        exprs = exprs.cons(e);
        }
      if(exprs.count() == 0)
        exprs = exprs.cons(NIL_EXPR);
      return new BodyExpr(exprs);
    }
  }

  public Object eval() throws Exception{
    Object ret = null;
    for(Object o : exprs)
      {
      Expr e = (Expr) o;
      ret = e.eval();
      }
    return ret;
  }

  public boolean canEmitPrimitive(){
    return lastExpr() instanceof MaybePrimitiveExpr && ((MaybePrimitiveExpr)lastExpr()).canEmitPrimitive();
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    for(int i = 0; i < exprs.count() - 1; i++)
      {
      Expr e = (Expr) exprs.nth(i);
      e.emit(C.STATEMENT, objx, gen);
      }
    MaybePrimitiveExpr last = (MaybePrimitiveExpr) exprs.nth(exprs.count() - 1);
    last.emitUnboxed(context, objx, gen);
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    for(int i = 0; i < exprs.count() - 1; i++)
      {
      Expr e = (Expr) exprs.nth(i);
      e.emit(C.STATEMENT, objx, gen);
      }
    Expr last = (Expr) exprs.nth(exprs.count() - 1);
    last.emit(context, objx, gen);
  }

  public boolean hasJavaClass() throws Exception{
    return lastExpr().hasJavaClass();
  }

  public Class getJavaClass() throws Exception{
    return lastExpr().getJavaClass();
  }

  private Expr lastExpr(){
    return (Expr) exprs.nth(exprs.count() - 1);
  }
}

public static class BindingInit{
  LocalBinding binding;
  Expr init;

  public final LocalBinding binding(){
    return binding;
  }

  public final Expr init(){
    return init;
  }

  public BindingInit(LocalBinding binding, Expr init){
    this.binding = binding;
    this.init = init;
  }
}

public static class LetFnExpr implements Expr{
  public final PersistentVector bindingInits;
  public final Expr body;

  public LetFnExpr(PersistentVector bindingInits, Expr body){
    this.bindingInits = bindingInits;
    this.body = body;
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object frm) throws Exception{
      ISeq form = (ISeq) frm;
      //(letfns* [var (fn [args] body) ...] body...)
      if(!(RT.second(form) instanceof IPersistentVector))
        throw new IllegalArgumentException("Bad binding form, expected vector");

      IPersistentVector bindings = (IPersistentVector) RT.second(form);
      if((bindings.count() % 2) != 0)
        throw new IllegalArgumentException("Bad binding form, expected matched symbol expression pairs");

      ISeq body = RT.next(RT.next(form));

      if(context == C.EVAL)
        return analyze(context, RT.list(RT.list(FN, PersistentVector.EMPTY, form)));

      IPersistentMap dynamicBindings = RT.map(LOCAL_ENV, LOCAL_ENV.deref(),
                                              NEXT_LOCAL_NUM, NEXT_LOCAL_NUM.deref());

      try
        {
        Var.pushThreadBindings(dynamicBindings);

        //pre-seed env (like Lisp labels)
        PersistentVector lbs = PersistentVector.EMPTY;
        for(int i = 0; i < bindings.count(); i += 2)
          {
          if(!(bindings.nth(i) instanceof Symbol))
            throw new IllegalArgumentException(
                "Bad binding form, expected symbol, got: " + bindings.nth(i));
          Symbol sym = (Symbol) bindings.nth(i);
          if(sym.getNamespace() != null)
            throw new Exception("Can't let qualified name: " + sym);
          LocalBinding lb = registerLocal(sym, tagOf(sym), null,false);
          lb.canBeCleared = false;
          lbs = lbs.cons(lb);
          }
        PersistentVector bindingInits = PersistentVector.EMPTY;
        for(int i = 0; i < bindings.count(); i += 2)
          {
          Symbol sym = (Symbol) bindings.nth(i);
          Expr init = analyze(C.EXPRESSION, bindings.nth(i + 1), sym.name);
          LocalBinding lb = (LocalBinding) lbs.nth(i / 2);
          lb.init = init;
          BindingInit bi = new BindingInit(lb, init);
          bindingInits = bindingInits.cons(bi);
          }
        return new LetFnExpr(bindingInits, (new BodyExpr.Parser()).parse(context, body));
        }
      finally
        {
        Var.popThreadBindings();
        }
    }
  }

  public Object eval() throws Exception{
    throw new UnsupportedOperationException("Can't eval letfns");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    for(int i = 0; i < bindingInits.count(); i++)
      {
      BindingInit bi = (BindingInit) bindingInits.nth(i);
      gen.visitInsn(Opcodes.ACONST_NULL);
      gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), bi.binding.idx);
      }

    IPersistentSet lbset = PersistentHashSet.EMPTY;

    for(int i = 0; i < bindingInits.count(); i++)
      {
      BindingInit bi = (BindingInit) bindingInits.nth(i);
      lbset = (IPersistentSet) lbset.cons(bi.binding);
      bi.init.emit(C.EXPRESSION, objx, gen);
      gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), bi.binding.idx);
      }

    for(int i = 0; i < bindingInits.count(); i++)
      {
      BindingInit bi = (BindingInit) bindingInits.nth(i);
      ObjExpr fe = (ObjExpr) bi.init;
      gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ILOAD), bi.binding.idx);
      fe.emitLetFnInits(gen, objx, lbset);
      }

    Label loopLabel = gen.mark();

    body.emit(context, objx, gen);

    Label end = gen.mark();
//    gen.visitLocalVariable("this", "Ljava/lang/Object;", null, loopLabel, end, 0);
    for(ISeq bis = bindingInits.seq(); bis != null; bis = bis.next())
      {
      BindingInit bi = (BindingInit) bis.first();
      String lname = bi.binding.name;
      if(lname.endsWith("__auto__"))
        lname += RT.nextID();
      Class primc = maybePrimitiveType(bi.init);
      if(primc != null)
        gen.visitLocalVariable(lname, Type.getDescriptor(primc), null, loopLabel, end,
                               bi.binding.idx);
      else
        gen.visitLocalVariable(lname, "Ljava/lang/Object;", null, loopLabel, end, bi.binding.idx);
      }
  }

  public boolean hasJavaClass() throws Exception{
    return body.hasJavaClass();
  }

  public Class getJavaClass() throws Exception{
    return body.getJavaClass();
  }
}

public static class LetExpr implements Expr, MaybePrimitiveExpr{
  public final PersistentVector bindingInits;
  public final Expr body;
  public final boolean isLoop;

  public LetExpr(PersistentVector bindingInits, Expr body, boolean isLoop){
    this.bindingInits = bindingInits;
    this.body = body;
    this.isLoop = isLoop;
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object frm) throws Exception{
      ISeq form = (ISeq) frm;
      //(let [var val var2 val2 ...] body...)
      boolean isLoop = RT.first(form).equals(LOOP);
      if(!(RT.second(form) instanceof IPersistentVector))
        throw new IllegalArgumentException("Bad binding form, expected vector");

      IPersistentVector bindings = (IPersistentVector) RT.second(form);
      if((bindings.count() % 2) != 0)
        throw new IllegalArgumentException("Bad binding form, expected matched symbol expression pairs");

      ISeq body = RT.next(RT.next(form));

      if(context == C.EVAL
         || (context == C.EXPRESSION && isLoop))
        return analyze(context, RT.list(RT.list(FN, PersistentVector.EMPTY, form)));

      IPersistentMap dynamicBindings = RT.map(LOCAL_ENV, LOCAL_ENV.deref(),
                                              NEXT_LOCAL_NUM, NEXT_LOCAL_NUM.deref());
      if(isLoop)
        dynamicBindings = dynamicBindings.assoc(LOOP_LOCALS, null);

      try
        {
        Var.pushThreadBindings(dynamicBindings);

        PersistentVector bindingInits = PersistentVector.EMPTY;
        PersistentVector loopLocals = PersistentVector.EMPTY;
        for(int i = 0; i < bindings.count(); i += 2)
          {
          if(!(bindings.nth(i) instanceof Symbol))
            throw new IllegalArgumentException(
                "Bad binding form, expected symbol, got: " + bindings.nth(i));
          Symbol sym = (Symbol) bindings.nth(i);
          if(sym.getNamespace() != null)
            throw new Exception("Can't let qualified name: " + sym);
          Expr init = analyze(C.EXPRESSION, bindings.nth(i + 1), sym.name);
          //sequential enhancement of env (like Lisp let*)
          LocalBinding lb = registerLocal(sym, tagOf(sym), init,false);
          BindingInit bi = new BindingInit(lb, init);
          bindingInits = bindingInits.cons(bi);

          if(isLoop)
            loopLocals = loopLocals.cons(lb);
          }
        if(isLoop)
          LOOP_LOCALS.set(loopLocals);
                Expr bodyExpr;
                try {
                    if(isLoop)
                        {
                        PathNode root = new PathNode(PATHTYPE.PATH, (PathNode) CLEAR_PATH.get());
                        Var.pushThreadBindings(
                            RT.map(CLEAR_PATH, new PathNode(PATHTYPE.PATH,root),
                                   CLEAR_ROOT, new PathNode(PATHTYPE.PATH,root)));
                        }
                    bodyExpr = (new BodyExpr.Parser()).parse(isLoop ? C.RETURN : context, body);
                    }
                finally{
                    if(isLoop)
                       Var.popThreadBindings();
                    }
        return new LetExpr(bindingInits, bodyExpr,
                           isLoop);
        }
      finally
        {
        Var.popThreadBindings();
        }
    }
  }

  public Object eval() throws Exception{
    throw new UnsupportedOperationException("Can't eval let/loop");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    doEmit(context, objx, gen, false);
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    doEmit(context, objx, gen, true);
  }


  public void doEmit(C context, ObjExpr objx, GeneratorAdapter gen, boolean emitUnboxed){
    for(int i = 0; i < bindingInits.count(); i++)
      {
      BindingInit bi = (BindingInit) bindingInits.nth(i);
      Class primc = maybePrimitiveType(bi.init);
      if(primc != null)
        {
        ((MaybePrimitiveExpr) bi.init).emitUnboxed(C.EXPRESSION, objx, gen);
        gen.visitVarInsn(Type.getType(primc).getOpcode(Opcodes.ISTORE), bi.binding.idx);
        }
      else
        {
        bi.init.emit(C.EXPRESSION, objx, gen);
        gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), bi.binding.idx);
        }
      }
    Label loopLabel = gen.mark();
    if(isLoop)
      {
      try
        {
        Var.pushThreadBindings(RT.map(LOOP_LABEL, loopLabel));
        if(emitUnboxed)
          ((MaybePrimitiveExpr)body).emitUnboxed(context, objx, gen);
        else
          body.emit(context, objx, gen);
        }
      finally
        {
        Var.popThreadBindings();
        }
      }
    else
      {
      if(emitUnboxed)
        ((MaybePrimitiveExpr)body).emitUnboxed(context, objx, gen);
      else
        body.emit(context, objx, gen);
      }
    Label end = gen.mark();
//    gen.visitLocalVariable("this", "Ljava/lang/Object;", null, loopLabel, end, 0);
    for(ISeq bis = bindingInits.seq(); bis != null; bis = bis.next())
      {
      BindingInit bi = (BindingInit) bis.first();
      String lname = bi.binding.name;
      if(lname.endsWith("__auto__"))
        lname += RT.nextID();
      Class primc = maybePrimitiveType(bi.init);
      if(primc != null)
        gen.visitLocalVariable(lname, Type.getDescriptor(primc), null, loopLabel, end,
                               bi.binding.idx);
      else
        gen.visitLocalVariable(lname, "Ljava/lang/Object;", null, loopLabel, end, bi.binding.idx);
      }
  }

  public boolean hasJavaClass() throws Exception{
    return body.hasJavaClass();
  }

  public Class getJavaClass() throws Exception{
    return body.getJavaClass();
  }

  public boolean canEmitPrimitive(){
    return body instanceof MaybePrimitiveExpr && ((MaybePrimitiveExpr)body).canEmitPrimitive();
  }

}

public static class RecurExpr implements Expr{
  public final IPersistentVector args;
  public final IPersistentVector loopLocals;

  public RecurExpr(IPersistentVector loopLocals, IPersistentVector args){
    this.loopLocals = loopLocals;
    this.args = args;
  }

  public Object eval() throws Exception{
    throw new UnsupportedOperationException("Can't eval recur");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    Label loopLabel = (Label) LOOP_LABEL.deref();
    if(loopLabel == null)
      throw new IllegalStateException();
    for(int i = 0; i < loopLocals.count(); i++)
      {
      LocalBinding lb = (LocalBinding) loopLocals.nth(i);
      Expr arg = (Expr) args.nth(i);
      if(lb.getPrimitiveType() != null)
        {
        Class primc = lb.getPrimitiveType();
        try
          {
          if(!(arg instanceof MaybePrimitiveExpr && arg.hasJavaClass() && arg.getJavaClass() == primc))
            throw new IllegalArgumentException("recur arg for primitive local: " +
                                               lb.name + " must be matching primitive");
          }
        catch(Exception e)
          {
          throw new RuntimeException(e);
          }
        ((MaybePrimitiveExpr) arg).emitUnboxed(C.EXPRESSION, objx, gen);
        }
      else
        {
        arg.emit(C.EXPRESSION, objx, gen);
        }
      }

    for(int i = loopLocals.count() - 1; i >= 0; i--)
      {
      LocalBinding lb = (LocalBinding) loopLocals.nth(i);
      Class primc = lb.getPrimitiveType();
      if(lb.isArg)
        gen.storeArg(lb.idx-1);
      else
        {
        if(primc != null)
          gen.visitVarInsn(Type.getType(primc).getOpcode(Opcodes.ISTORE), lb.idx);
        else
          gen.visitVarInsn(OBJECT_TYPE.getOpcode(Opcodes.ISTORE), lb.idx);
        }
      }

    gen.goTo(loopLabel);
  }

  public boolean hasJavaClass() throws Exception{
    return true;
  }

  public Class getJavaClass() throws Exception{
    return null;
  }

  static class Parser implements IParser{
    public Expr parse(C context, Object frm) throws Exception{
      ISeq form = (ISeq) frm;
      IPersistentVector loopLocals = (IPersistentVector) LOOP_LOCALS.deref();
      if(context != C.RETURN || loopLocals == null)
        throw new UnsupportedOperationException("Can only recur from tail position");
      if(IN_CATCH_FINALLY.deref() != null)
        throw new UnsupportedOperationException("Cannot recur from catch/finally");
      PersistentVector args = PersistentVector.EMPTY;
      for(ISeq s = RT.seq(form.next()); s != null; s = s.next())
        {
        args = args.cons(analyze(C.EXPRESSION, s.first()));
        }
      if(args.count() != loopLocals.count())
        throw new IllegalArgumentException(
            String.format("Mismatched argument count to recur, expected: %d args, got: %d",
                          loopLocals.count(), args.count()));
      return new RecurExpr(loopLocals, args);
    }
  }
}

private static LocalBinding registerLocal(Symbol sym, Symbol tag, Expr init, boolean isArg) throws Exception{
  int num = getAndIncLocalNum();
  LocalBinding b = new LocalBinding(num, sym, tag, init, isArg, clearPathRoot());
  IPersistentMap localsMap = (IPersistentMap) LOCAL_ENV.deref();
  LOCAL_ENV.set(RT.assoc(localsMap, b.sym, b));
  ObjMethod method = (ObjMethod) METHOD.deref();
  method.locals = (IPersistentMap) RT.assoc(method.locals, b, b);
  method.indexlocals = (IPersistentMap) RT.assoc(method.indexlocals, num, b);
  return b;
}

private static int getAndIncLocalNum(){
  int num = ((Number) NEXT_LOCAL_NUM.deref()).intValue();
  ObjMethod m = (ObjMethod) METHOD.deref();
  if(num > m.maxLocal)
    m.maxLocal = num;
  NEXT_LOCAL_NUM.set(num + 1);
  return num;
}

public static Expr analyze(C context, Object form) throws Exception{
  return analyze(context, form, null);
}

private static Expr analyze(C context, Object form, String name) throws Exception{
  //todo symbol macro expansion?
  try
    {
    if(form instanceof LazySeq)
      {
      form = RT.seq(form);
      if(form == null)
        form = PersistentList.EMPTY;
      }
    if(form == null)
      return NIL_EXPR;
    else if(form == Boolean.TRUE)
      return TRUE_EXPR;
    else if(form == Boolean.FALSE)
        return FALSE_EXPR;
    Class fclass = form.getClass();
    if(fclass == Symbol.class)
      return analyzeSymbol((Symbol) form);
    else if(fclass == Keyword.class)
      return registerKeyword((Keyword) form);
//  else if(form instanceof Num)
//    return new NumExpr((Num) form);
    else if(fclass == String.class)
        return new StringExpr(((String) form).intern());
//  else if(fclass == Character.class)
//    return new CharExpr((Character) form);
      else if(form instanceof IPersistentCollection && ((IPersistentCollection) form).count() == 0)
          {
          Expr ret = new EmptyExpr(form);
          if(RT.meta(form) != null)
            ret = new MetaExpr(ret, (MapExpr) MapExpr
                .parse(context == C.EVAL ? context : C.EXPRESSION, ((IObj) form).meta()));
          return ret;
          }
        else if(form instanceof ISeq)
            return analyzeSeq(context, (ISeq) form, name);
          else if(form instanceof IPersistentVector)
              return VectorExpr.parse(context, (IPersistentVector) form);
            else if(form instanceof IPersistentMap)
                return MapExpr.parse(context, (IPersistentMap) form);
              else if(form instanceof IPersistentSet)
                  return SetExpr.parse(context, (IPersistentSet) form);

//  else
    //throw new UnsupportedOperationException();
    return new ConstantExpr(form);
    }
  catch(Throwable e)
    {
    if(!(e instanceof CompilerException))
      throw new CompilerException((String) SOURCE.deref(), (Integer) LINE.deref(), e);
    else
      throw (CompilerException) e;
    }
}

static public class CompilerException extends Exception{

  public CompilerException(String source, int line, Throwable cause){
    super(errorMsg(source, line, cause.toString()), cause);
  }

  public String toString(){
    return getMessage();
  }
}

static public Var isMacro(Object op) throws Exception{
  //no local macros for now
  if(op instanceof Symbol && referenceLocal((Symbol) op) != null)
    return null;
  if(op instanceof Symbol || op instanceof Var)
    {
    Var v = (op instanceof Var) ? (Var) op : lookupVar((Symbol) op, false);
    if(v != null && v.isMacro())
      {
      if(v.ns != currentNS() && !v.isPublic())
        throw new IllegalStateException("var: " + v + " is not public");
      return v;
      }
    }
  return null;
}

static public IFn isInline(Object op, int arity) throws Exception{
  //no local inlines for now
  if(op instanceof Symbol && referenceLocal((Symbol) op) != null)
    return null;
  if(op instanceof Symbol || op instanceof Var)
    {
    Var v = (op instanceof Var) ? (Var) op : lookupVar((Symbol) op, false);
    if(v != null)
      {
      if(v.ns != currentNS() && !v.isPublic())
        throw new IllegalStateException("var: " + v + " is not public");
      IFn ret = (IFn) RT.get(v.meta(), inlineKey);
      if(ret != null)
        {
        IFn arityPred = (IFn) RT.get(v.meta(), inlineAritiesKey);
        if(arityPred == null || RT.booleanCast(arityPred.invoke(arity)))
          return ret;
        }
      }
    }
  return null;
}

public static boolean namesStaticMember(Symbol sym){
  return sym.ns != null && namespaceFor(sym) == null;
}

public static Object preserveTag(ISeq src, Object dst) {
  Symbol tag = tagOf(src);
  if (tag != null && dst instanceof IObj) {
    IPersistentMap meta = RT.meta(dst);
    return ((IObj) dst).withMeta((IPersistentMap) RT.assoc(meta, RT.TAG_KEY, tag));
  }
  return dst;
}

public static Object macroexpand1(Object x) throws Exception{
  if(x instanceof ISeq)
    {
    ISeq form = (ISeq) x;
    Object op = RT.first(form);
    if(isSpecial(op))
      return x;
    //macro expansion
    Var v = isMacro(op);
    if(v != null)
      {
      return v.applyTo(RT.cons(form,RT.cons(LOCAL_ENV.get(),form.next())));
      }
    else
      {
      if(op instanceof Symbol)
        {
        Symbol sym = (Symbol) op;
        String sname = sym.name;
        //(.substring s 2 5) => (. s substring 2 5)
        if(sym.name.charAt(0) == '.')
          {
          if(RT.length(form) < 2)
            throw new IllegalArgumentException(
                "Malformed member expression, expecting (.member target ...)");
          Symbol meth = Symbol.intern(sname.substring(1));
          Object target = RT.second(form);
          if(HostExpr.maybeClass(target, false) != null)
            {
            target = ((IObj)RT.list(IDENTITY, target)).withMeta(RT.map(RT.TAG_KEY,CLASS));
            }
          return preserveTag(form, RT.listStar(DOT, target, meth, form.next().next()));
          }
        else if(namesStaticMember(sym))
          {
          Symbol target = Symbol.intern(sym.ns);
          Class c = HostExpr.maybeClass(target, false);
          if(c != null)
            {
            Symbol meth = Symbol.intern(sym.name);
            return preserveTag(form, RT.listStar(DOT, target, meth, form.next()));
            }
          }
        else
          {
          //(s.substring 2 5) => (. s substring 2 5)
          //also (package.class.name ...) (. package.class name ...)
          int idx = sname.lastIndexOf('.');
//          if(idx > 0 && idx < sname.length() - 1)
//            {
//            Symbol target = Symbol.intern(sname.substring(0, idx));
//            Symbol meth = Symbol.intern(sname.substring(idx + 1));
//            return RT.listStar(DOT, target, meth, form.rest());
//            }
          //(StringBuilder. "foo") => (new StringBuilder "foo") 
          //else
          if(idx == sname.length() - 1)
            return RT.listStar(NEW, Symbol.intern(sname.substring(0, idx)), form.next());
          }
        }
      }
    }
  return x;
}

static Object macroexpand(Object form) throws Exception{
  Object exf = macroexpand1(form);
  if(exf != form)
    return macroexpand(exf);
  return form;
}

private static Expr analyzeSeq(C context, ISeq form, String name) throws Exception{
  Integer line = (Integer) LINE.deref();
  if(RT.meta(form) != null && RT.meta(form).containsKey(RT.LINE_KEY))
    line = (Integer) RT.meta(form).valAt(RT.LINE_KEY);
  Var.pushThreadBindings(
      RT.map(LINE, line));
  try
    {
    Object me = macroexpand1(form);
    if(me != form)
      return analyze(context, me, name);

    Object op = RT.first(form);
    if(op == null)
      throw new IllegalArgumentException("Can't call nil");
    IFn inline = isInline(op, RT.count(RT.next(form)));
    if(inline != null)
      return analyze(context, preserveTag(form, inline.applyTo(RT.next(form))));
    IParser p;
    if(op.equals(FN))
      return FnExpr.parse(context, form, name);
    else if((p = (IParser) specials.valAt(op)) != null)
      return p.parse(context, form);
    else
      return InvokeExpr.parse(context, form);
    }
  catch(Throwable e)
    {
    if(!(e instanceof CompilerException))
      throw new CompilerException((String) SOURCE.deref(), (Integer) LINE.deref(), e);
    else
      throw (CompilerException) e;
    }
  finally
    {
    Var.popThreadBindings();
    }
}

static String errorMsg(String source, int line, String s){
  return String.format("%s (%s:%d)", s, source, line);
}

public static Object eval(Object form) throws Exception{
  return eval(form, true);
}

public static Object eval(Object form, boolean freshLoader) throws Exception{
  boolean createdLoader = false;
  if(true)//!LOADER.isBound())
    {
    Var.pushThreadBindings(RT.map(LOADER, RT.makeClassLoader()));
    createdLoader = true;
    }
  try
    {
    Integer line = (Integer) LINE.deref();
    if(RT.meta(form) != null && RT.meta(form).containsKey(RT.LINE_KEY))
      line = (Integer) RT.meta(form).valAt(RT.LINE_KEY);
    Var.pushThreadBindings(RT.map(LINE, line));
    try
      {
      form = macroexpand(form);
      if(form instanceof IPersistentCollection && Util.equals(RT.first(form), DO))
        {
        ISeq s = RT.next(form);
        for(; RT.next(s) != null; s = RT.next(s))
          eval(RT.first(s),false);
        return eval(RT.first(s),false);
        }
      else if(form instanceof IPersistentCollection
              && !(RT.first(form) instanceof Symbol
                   && ((Symbol) RT.first(form)).name.startsWith("def")))
        {
        ObjExpr fexpr = (ObjExpr) analyze(C.EXPRESSION, RT.list(FN, PersistentVector.EMPTY, form),
                                          "eval" + RT.nextID());
        IFn fn = (IFn) fexpr.eval();
        return fn.invoke();
        }
      else
        {
        Expr expr = analyze(C.EVAL, form);
        return expr.eval();
        }
      }
    finally
      {
      Var.popThreadBindings();
      }
    }
  catch(Throwable e)
    {
    if(!(e instanceof CompilerException))
      throw new CompilerException((String) SOURCE.deref(), (Integer) LINE.deref(), e);
    else
      throw (CompilerException) e;
    }
  finally
    {
    if(createdLoader)
      Var.popThreadBindings();
    }
}

private static int registerConstant(Object o){
  if(!CONSTANTS.isBound())
    return -1;
  PersistentVector v = (PersistentVector) CONSTANTS.deref();
  IdentityHashMap<Object,Integer> ids = (IdentityHashMap<Object,Integer>) CONSTANT_IDS.deref();
  Integer i = ids.get(o);
  if(i != null)
    return i;
  CONSTANTS.set(RT.conj(v, o));
  ids.put(o, v.count());
  return v.count();
}

private static KeywordExpr registerKeyword(Keyword keyword){
  if(!KEYWORDS.isBound())
    return new KeywordExpr(keyword);

  IPersistentMap keywordsMap = (IPersistentMap) KEYWORDS.deref();
  Object id = RT.get(keywordsMap, keyword);
  if(id == null)
    {
    KEYWORDS.set(RT.assoc(keywordsMap, keyword, registerConstant(keyword)));
    }
  return new KeywordExpr(keyword);
//  KeywordExpr ke = (KeywordExpr) RT.get(keywordsMap, keyword);
//  if(ke == null)
//    KEYWORDS.set(RT.assoc(keywordsMap, keyword, ke = new KeywordExpr(keyword)));
//  return ke;
}

private static int registerKeywordCallsite(Keyword keyword){
  if(!KEYWORD_CALLSITES.isBound())
    throw new IllegalAccessError("KEYWORD_CALLSITES is not bound");

  IPersistentVector keywordCallsites = (IPersistentVector) KEYWORD_CALLSITES.deref();

  keywordCallsites = keywordCallsites.cons(keyword);
  KEYWORD_CALLSITES.set(keywordCallsites);
  return keywordCallsites.count()-1;
}

private static int registerProtocolCallsite(Var v){
  if(!PROTOCOL_CALLSITES.isBound())
    throw new IllegalAccessError("PROTOCOL_CALLSITES is not bound");

  IPersistentVector protocolCallsites = (IPersistentVector) PROTOCOL_CALLSITES.deref();

  protocolCallsites = protocolCallsites.cons(v);
  PROTOCOL_CALLSITES.set(protocolCallsites);
  return protocolCallsites.count()-1;
}

private static int registerVarCallsite(Var v){
  if(!VAR_CALLSITES.isBound())
    throw new IllegalAccessError("VAR_CALLSITES is not bound");

  IPersistentVector varCallsites = (IPersistentVector) VAR_CALLSITES.deref();

  varCallsites = varCallsites.cons(v);
  VAR_CALLSITES.set(varCallsites);
  return varCallsites.count()-1;
}

static ISeq fwdPath(PathNode p1){
    ISeq ret = null;
    for(;p1 != null;p1 = p1.parent)
        ret = RT.cons(p1,ret);
    return ret;
}

static PathNode commonPath(PathNode n1, PathNode n2){
    ISeq xp = fwdPath(n1);
    ISeq yp = fwdPath(n2);
    if(RT.first(xp) != RT.first(yp))
        return null;
    while(RT.second(xp) != null && RT.second(xp) == RT.second(yp))
        {
        xp = xp.next();
        yp = yp.next();
        }
    return (PathNode) RT.first(xp);
}

static void addAnnotation(Object visitor, IPersistentMap meta){
  try{
  if(meta != null && ADD_ANNOTATIONS.isBound())
     ADD_ANNOTATIONS.invoke(visitor, meta);
  }
  catch (Exception e)
    {
    throw new RuntimeException(e);
    }
}

static void addParameterAnnotation(Object visitor, IPersistentMap meta, int i){
  try{
  if(meta != null && ADD_ANNOTATIONS.isBound())
     ADD_ANNOTATIONS.invoke(visitor, meta, i);
  }
  catch (Exception e)
    {
    throw new RuntimeException(e);
    }
}

private static Expr analyzeSymbol(Symbol sym) throws Exception{
  Symbol tag = tagOf(sym);
  if(sym.ns == null) //ns-qualified syms are always Vars
    {
    LocalBinding b = referenceLocal(sym);
    if(b != null)
            {
            return new LocalBindingExpr(b, tag);
            }
    }
  else
    {
    if(namespaceFor(sym) == null)
      {
      Symbol nsSym = Symbol.create(sym.ns);
      Class c = HostExpr.maybeClass(nsSym, false);
      if(c != null)
        {
        if(Reflector.getField(c, sym.name, true) != null)
          return new StaticFieldExpr((Integer) LINE.deref(), c, sym.name, tag);
        throw new Exception("Unable to find static field: " + sym.name + " in " + c);
        }
      }
    }
  //Var v = lookupVar(sym, false);
//  Var v = lookupVar(sym, false);
//  if(v != null)
//    return new VarExpr(v, tag);
  Object o = resolve(sym);
  if(o instanceof Var)
    {
    Var v = (Var) o;
    if(isMacro(v) != null)
      throw new Exception("Can't take value of a macro: " + v);
    registerVar(v);
    return new VarExpr(v, tag);
    }
  else if(o instanceof Class)
    return new ConstantExpr(o);
  else if(o instanceof Symbol)
      return new UnresolvedVarExpr((Symbol) o);

  throw new Exception("Unable to resolve symbol: " + sym + " in this context");

}

static String destubClassName(String className){
  //skip over prefix + '.' or '/'
  if(className.startsWith(COMPILE_STUB_PREFIX))
    return className.substring(COMPILE_STUB_PREFIX.length()+1);
  return className;
}

static Type getType(Class c){
  String descriptor = Type.getType(c).getDescriptor();
  if(descriptor.startsWith("L"))
    descriptor = "L" + destubClassName(descriptor.substring(1));
  return Type.getType(descriptor);
}

static Object resolve(Symbol sym, boolean allowPrivate) throws Exception{
  return resolveIn(currentNS(), sym, allowPrivate);
}

static Object resolve(Symbol sym) throws Exception{
  return resolveIn(currentNS(), sym, false);
}

static Namespace namespaceFor(Symbol sym){
  return namespaceFor(currentNS(), sym);
}

static Namespace namespaceFor(Namespace inns, Symbol sym){
  //note, presumes non-nil sym.ns
  // first check against currentNS' aliases...
  Symbol nsSym = Symbol.create(sym.ns);
  Namespace ns = inns.lookupAlias(nsSym);
  if(ns == null)
    {
    // ...otherwise check the Namespaces map.
    ns = Namespace.find(nsSym);
    }
  return ns;
}

static public Object resolveIn(Namespace n, Symbol sym, boolean allowPrivate) throws Exception{
  //note - ns-qualified vars must already exist
  if(sym.ns != null)
    {
    Namespace ns = namespaceFor(n, sym);
    if(ns == null)
      throw new Exception("No such namespace: " + sym.ns);

    Var v = ns.findInternedVar(Symbol.create(sym.name));
    if(v == null)
      throw new Exception("No such var: " + sym);
    else if(v.ns != currentNS() && !v.isPublic() && !allowPrivate)
      throw new IllegalStateException("var: " + sym + " is not public");
    return v;
    }
  else if(sym.name.indexOf('.') > 0 || sym.name.charAt(0) == '[')
    {
    return RT.classForName(sym.name);
    }
  else if(sym.equals(NS))
      return RT.NS_VAR;
    else if(sym.equals(IN_NS))
        return RT.IN_NS_VAR;
      else
        {
        if(Util.equals(sym,COMPILE_STUB_SYM.get()))
          return COMPILE_STUB_CLASS.get();
        Object o = n.getMapping(sym);
        if(o == null)
          {
          if(RT.booleanCast(RT.ALLOW_UNRESOLVED_VARS.deref()))
            {
            return sym;
            }
          else
            {
            throw new Exception("Unable to resolve symbol: " + sym + " in this context");
            }
          }
        return o;
        }
}


static public Object maybeResolveIn(Namespace n, Symbol sym) throws Exception{
  //note - ns-qualified vars must already exist
  if(sym.ns != null)
    {
    Namespace ns = namespaceFor(n, sym);
    if(ns == null)
      return null;
    Var v = ns.findInternedVar(Symbol.create(sym.name));
    if(v == null)
      return null;
    return v;
    }
  else if(sym.name.indexOf('.') > 0 && !sym.name.endsWith(".")
      || sym.name.charAt(0) == '[')
    {
    return RT.classForName(sym.name);
    }
  else if(sym.equals(NS))
      return RT.NS_VAR;
    else if(sym.equals(IN_NS))
        return RT.IN_NS_VAR;
      else
        {
        Object o = n.getMapping(sym);
        return o;
        }
}


static Var lookupVar(Symbol sym, boolean internNew) throws Exception{
  Var var = null;

  //note - ns-qualified vars in other namespaces must already exist
  if(sym.ns != null)
    {
    Namespace ns = namespaceFor(sym);
    if(ns == null)
      return null;
    //throw new Exception("No such namespace: " + sym.ns);
    Symbol name = Symbol.create(sym.name);
    if(internNew && ns == currentNS())
      var = currentNS().intern(name);
    else
      var = ns.findInternedVar(name);
    }
  else if(sym.equals(NS))
    var = RT.NS_VAR;
  else if(sym.equals(IN_NS))
      var = RT.IN_NS_VAR;
    else
      {
      //is it mapped?
      Object o = currentNS().getMapping(sym);
      if(o == null)
        {
        //introduce a new var in the current ns
        if(internNew)
          var = currentNS().intern(Symbol.create(sym.name));
        }
      else if(o instanceof Var)
        {
        var = (Var) o;
        }
      else
        {
        throw new Exception("Expecting var, but " + sym + " is mapped to " + o);
        }
      }
  if(var != null)
    registerVar(var);
  return var;
}

private static void registerVar(Var var) throws Exception{
  if(!VARS.isBound())
    return;
  IPersistentMap varsMap = (IPersistentMap) VARS.deref();
  Object id = RT.get(varsMap, var);
  if(id == null)
    {
    VARS.set(RT.assoc(varsMap, var, registerConstant(var)));
    }
//  if(varsMap != null && RT.get(varsMap, var) == null)
//    VARS.set(RT.assoc(varsMap, var, var));
}

static Namespace currentNS(){
  return (Namespace) RT.CURRENT_NS.deref();
}

static void closeOver(LocalBinding b, ObjMethod method){
  if(b != null && method != null)
    {
    if(RT.get(method.locals, b) == null)
      {
      method.objx.closes = (IPersistentMap) RT.assoc(method.objx.closes, b, b);
      closeOver(b, method.parent);
      }
    else if(IN_CATCH_FINALLY.deref() != null)
      {
      method.localsUsedInCatchFinally = (PersistentHashSet) method.localsUsedInCatchFinally.cons(b.idx);
      }
    }
}


static LocalBinding referenceLocal(Symbol sym) throws Exception{
  if(!LOCAL_ENV.isBound())
    return null;
  LocalBinding b = (LocalBinding) RT.get(LOCAL_ENV.deref(), sym);
  if(b != null)
    {
    ObjMethod method = (ObjMethod) METHOD.deref();
    closeOver(b, method);
    }
  return b;
}

private static Symbol tagOf(Object o){
  Object tag = RT.get(RT.meta(o), RT.TAG_KEY);
  if(tag instanceof Symbol)
    return (Symbol) tag;
  else if(tag instanceof String)
    return Symbol.intern(null, (String) tag);
  return null;
}

public static Object loadFile(String file) throws Exception{
//  File fo = new File(file);
//  if(!fo.exists())
//    return null;

  FileInputStream f = new FileInputStream(file);
  try
    {
    return load(new InputStreamReader(f, RT.UTF8), new File(file).getAbsolutePath(), (new File(file)).getName());
    }
  finally
    {
    f.close();
    }
}

public static Object load(Reader rdr) throws Exception{
  return load(rdr, null, "NO_SOURCE_FILE");
}

public static Object load(Reader rdr, String sourcePath, String sourceName) throws Exception{
  Object EOF = new Object();
  Object ret = null;
  LineNumberingPushbackReader pushbackReader =
      (rdr instanceof LineNumberingPushbackReader) ? (LineNumberingPushbackReader) rdr :
      new LineNumberingPushbackReader(rdr);
  Var.pushThreadBindings(
      RT.map(LOADER, RT.makeClassLoader(),
             SOURCE_PATH, sourcePath,
             SOURCE, sourceName,
             METHOD, null,
             LOCAL_ENV, null,
          LOOP_LOCALS, null,
          NEXT_LOCAL_NUM, 0,
             RT.CURRENT_NS, RT.CURRENT_NS.deref(),
             LINE_BEFORE, pushbackReader.getLineNumber(),
             LINE_AFTER, pushbackReader.getLineNumber()
      ));

  try
    {
    for(Object r = LispReader.read(pushbackReader, false, EOF, false); r != EOF;
        r = LispReader.read(pushbackReader, false, EOF, false))
      {
      LINE_AFTER.set(pushbackReader.getLineNumber());
      ret = eval(r,false);
      LINE_BEFORE.set(pushbackReader.getLineNumber());
      }
    }
  catch(LispReader.ReaderException e)
    {
    throw new CompilerException(sourceName, e.line, e.getCause());
    }
  finally
    {
    Var.popThreadBindings();
    }
  return ret;
}

static public void writeClassFile(String internalName, byte[] bytecode) throws Exception{
  String genPath = (String) COMPILE_PATH.deref();
  if(genPath == null)
    throw new Exception("*compile-path* not set");
  String[] dirs = internalName.split("/");
  String p = genPath;
  for(int i = 0; i < dirs.length - 1; i++)
    {
    p += File.separator + dirs[i];
    (new File(p)).mkdir();
    }
  String path = genPath + File.separator + internalName + ".class";
  File cf = new File(path);
  cf.createNewFile();
  FileOutputStream cfs = new FileOutputStream(cf);
  try
    {
    cfs.write(bytecode);
    cfs.flush();
    cfs.getFD().sync();
    }
  finally
    {
    cfs.close();
    }
}

public static void pushNS(){
  Var.pushThreadBindings(PersistentHashMap.create(Var.intern(Symbol.create("clojure.core"),
                                                             Symbol.create("*ns*")), null));
}

public static ILookupThunk getLookupThunk(Object target, Keyword k){
  return null//To change body of created methods use File | Settings | File Templates.
}

static void compile1(GeneratorAdapter gen, ObjExpr objx, Object form) throws Exception{
  Integer line = (Integer) LINE.deref();
  if(RT.meta(form) != null && RT.meta(form).containsKey(RT.LINE_KEY))
    line = (Integer) RT.meta(form).valAt(RT.LINE_KEY);
  Var.pushThreadBindings(
      RT.map(LINE, line
             ,LOADER, RT.makeClassLoader()
      ));
  try
    {
    form = macroexpand(form);
    if(form instanceof IPersistentCollection && Util.equals(RT.first(form), DO))
      {
      for(ISeq s = RT.next(form); s != null; s = RT.next(s))
        {
        compile1(gen, objx, RT.first(s));
        }
      }
    else
      {
      Expr expr = analyze(C.EVAL, form);
      objx.keywords = (IPersistentMap) KEYWORDS.deref();
      objx.vars = (IPersistentMap) VARS.deref();
      objx.constants = (PersistentVector) CONSTANTS.deref();
      expr.emit(C.EXPRESSION, objx, gen);
      expr.eval();
      }
    }
  finally
    {
    Var.popThreadBindings();
    }
}

public static Object compile(Reader rdr, String sourcePath, String sourceName) throws Exception{
  if(COMPILE_PATH.deref() == null)
    throw new Exception("*compile-path* not set");

  Object EOF = new Object();
  Object ret = null;
  LineNumberingPushbackReader pushbackReader =
      (rdr instanceof LineNumberingPushbackReader) ? (LineNumberingPushbackReader) rdr :
      new LineNumberingPushbackReader(rdr);
  Var.pushThreadBindings(
      RT.map(SOURCE_PATH, sourcePath,
             SOURCE, sourceName,
             METHOD, null,
             LOCAL_ENV, null,
          LOOP_LOCALS, null,
          NEXT_LOCAL_NUM, 0,
             RT.CURRENT_NS, RT.CURRENT_NS.deref(),
             LINE_BEFORE, pushbackReader.getLineNumber(),
             LINE_AFTER, pushbackReader.getLineNumber(),
             CONSTANTS, PersistentVector.EMPTY,
             CONSTANT_IDS, new IdentityHashMap(),
             KEYWORDS, PersistentHashMap.EMPTY,
             VARS, PersistentHashMap.EMPTY
         //    ,LOADER, RT.makeClassLoader()
      ));

  try
    {
    //generate loader class
    ObjExpr objx = new ObjExpr(null);
    objx.internalName = sourcePath.replace(File.separator, "/").substring(0, sourcePath.lastIndexOf('.'))
                      + RT.LOADER_SUFFIX;

    objx.objtype = Type.getObjectType(objx.internalName);
    ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
    ClassVisitor cv = cw;
    cv.visit(V1_5, ACC_PUBLIC + ACC_SUPER, objx.internalName, null, "java/lang/Object", null);

    //static load method
    GeneratorAdapter gen = new GeneratorAdapter(ACC_PUBLIC + ACC_STATIC,
                                                Method.getMethod("void load ()"),
                                                null,
                                                null,
                                                cv);
    gen.visitCode();

    for(Object r = LispReader.read(pushbackReader, false, EOF, false); r != EOF;
        r = LispReader.read(pushbackReader, false, EOF, false))
      {
        LINE_AFTER.set(pushbackReader.getLineNumber());
        compile1(gen, objx, r);
        LINE_BEFORE.set(pushbackReader.getLineNumber());
      }
    //end of load
    gen.returnValue();
    gen.endMethod();

    //static fields for constants
    for(int i = 0; i < objx.constants.count(); i++)
      {
      cv.visitField(ACC_PUBLIC + ACC_FINAL + ACC_STATIC, objx.constantName(i), objx.constantType(i).getDescriptor(),
                    null, null);
      }

    //static init for constants, keywords and vars
    GeneratorAdapter clinitgen = new GeneratorAdapter(ACC_PUBLIC + ACC_STATIC,
                                                      Method.getMethod("void <clinit> ()"),
                                                      null,
                                                      null,
                                                      cv);
    clinitgen.visitCode();
    Label startTry = clinitgen.newLabel();
    Label endTry = clinitgen.newLabel();
    Label end = clinitgen.newLabel();
    Label finallyLabel = clinitgen.newLabel();

    if(objx.constants.count() > 0)
      {
      objx.emitConstants(clinitgen);
      }
    clinitgen.invokeStatic(Type.getType(Compiler.class), Method.getMethod("void pushNS()"));
    clinitgen.mark(startTry);
    clinitgen.invokeStatic(objx.objtype, Method.getMethod("void load()"));
    clinitgen.mark(endTry);
    clinitgen.invokeStatic(VAR_TYPE, Method.getMethod("void popThreadBindings()"));
    clinitgen.goTo(end);

    clinitgen.mark(finallyLabel);
    //exception should be on stack
    clinitgen.invokeStatic(VAR_TYPE, Method.getMethod("void popThreadBindings()"));
    clinitgen.throwException();
    clinitgen.mark(end);
    clinitgen.visitTryCatchBlock(startTry, endTry, finallyLabel, null);

    //end of static init
    clinitgen.returnValue();
    clinitgen.endMethod();

    //end of class
    cv.visitEnd();

    writeClassFile(objx.internalName, cw.toByteArray());
    }
  catch(LispReader.ReaderException e)
    {
    throw new CompilerException(sourceName, e.line, e.getCause());
    }
  finally
    {
    Var.popThreadBindings();
    }
  return ret;
}


static public class NewInstanceExpr extends ObjExpr{
  //IPersistentMap optionsMap = PersistentArrayMap.EMPTY;
  IPersistentCollection methods;

  Map<IPersistentVector,java.lang.reflect.Method> mmap;
  Map<IPersistentVector,Set<Class>> covariants;

  public NewInstanceExpr(Object tag){
    super(tag);
  }

  static class DeftypeParser implements IParser{
    public Expr parse(C context, final Object frm) throws Exception{
      ISeq rform = (ISeq) frm;
      //(deftype* tagname classname [fields] :implements [interfaces] :tag tagname methods*)
      rform = RT.next(rform);
      String tagname = ((Symbol) rform.first()).toString();
      rform = rform.next();
      Symbol classname = (Symbol) rform.first();
      rform = rform.next();
      IPersistentVector fields = (IPersistentVector) rform.first();
      rform = rform.next();
      IPersistentMap opts = PersistentHashMap.EMPTY;
      while(rform != null && rform.first() instanceof Keyword)
        {
        opts = opts.assoc(rform.first(), RT.second(rform));
        rform = rform.next().next();
        }

      ObjExpr ret = build((IPersistentVector)RT.get(opts,implementsKey,PersistentVector.EMPTY),fields,null,tagname, classname,
                   (Symbol) RT.get(opts,RT.TAG_KEY),rform, frm);
      return ret;
    }
  }

  static class ReifyParser implements IParser{
  public Expr parse(C context, Object frm) throws Exception{
    //(reify this-name? [interfaces] (method-name [args] body)*)
    ISeq form = (ISeq) frm;
    ObjMethod enclosingMethod = (ObjMethod) METHOD.deref();
    String basename = enclosingMethod != null ?
                      (trimGenID(enclosingMethod.objx.name) + "$")
                     : (munge(currentNS().name.name) + "$");
    String simpleName = "reify__" + RT.nextID();
    String classname = basename + simpleName;

    ISeq rform = RT.next(form);

    IPersistentVector interfaces = ((IPersistentVector) RT.first(rform)).cons(Symbol.intern("clojure.lang.IObj"));


    rform = RT.next(rform);


    ObjExpr ret = build(interfaces, null, null, classname, Symbol.intern(classname), null, rform, frm);
    if(frm instanceof IObj && ((IObj) frm).meta() != null)
      return new MetaExpr(ret, (MapExpr) MapExpr
          .parse(context == C.EVAL ? context : C.EXPRESSION, ((IObj) frm).meta()));
    else
      return ret;
  }
  }

  static ObjExpr build(IPersistentVector interfaceSyms, IPersistentVector fieldSyms, Symbol thisSym,
                       String tagName, Symbol className,
                    Symbol typeTag, ISeq methodForms, Object frm) throws Exception{
    NewInstanceExpr ret = new NewInstanceExpr(null);

    ret.src = frm;
    ret.name = className.toString();
    ret.classMeta = RT.meta(className);
    ret.internalName = ret.name.replace('.', '/');
    ret.objtype = Type.getObjectType(ret.internalName);

    if(thisSym != null)
      ret.thisName = thisSym.name;

    if(fieldSyms != null)
      {
      IPersistentMap fmap = PersistentHashMap.EMPTY;
      Object[] closesvec = new Object[2 * fieldSyms.count()];
      for(int i=0;i<fieldSyms.count();i++)
        {
        Symbol sym = (Symbol) fieldSyms.nth(i);
        LocalBinding lb = new LocalBinding(-1, sym, null,
                                           new MethodParamExpr(tagClass(tagOf(sym))),false,null);
        fmap = fmap.assoc(sym, lb);
        closesvec[i*2] = lb;
        closesvec[i*2 + 1] = lb;
        }

      //todo - inject __meta et al into closes - when?
      //use array map to preserve ctor order
      ret.closes = new PersistentArrayMap(closesvec);
      ret.fields = fmap;
      for(int i=fieldSyms.count()-1;i >= 0 && ((Symbol)fieldSyms.nth(i)).name.startsWith("__");--i)
        ret.altCtorDrops++;
      }
    //todo - set up volatiles
//    ret.volatiles = PersistentHashSet.create(RT.seq(RT.get(ret.optionsMap, volatileKey)));

    PersistentVector interfaces = PersistentVector.EMPTY;
    for(ISeq s = RT.seq(interfaceSyms);s!=null;s = s.next())
      {
      Class c = (Class) resolve((Symbol) s.first());
      if(!c.isInterface())
        throw new IllegalArgumentException("only interfaces are supported, had: " + c.getName());
      interfaces = interfaces.cons(c);
      }
    Class superClass = Object.class;
    Map[] mc = gatherMethods(superClass,RT.seq(interfaces));
    Map overrideables = mc[0];
    Map covariants = mc[1];
    ret.mmap = overrideables;
    ret.covariants = covariants;
   
    String[] inames = interfaceNames(interfaces);

    Class stub = compileStub(slashname(superClass),ret, inames, frm);
    Symbol thistag = Symbol.intern(null,stub.getName());

    try
      {
      Var.pushThreadBindings(
          RT.map(CONSTANTS, PersistentVector.EMPTY,
                 CONSTANT_IDS, new IdentityHashMap(),
                 KEYWORDS, PersistentHashMap.EMPTY,
                 VARS, PersistentHashMap.EMPTY,
                 KEYWORD_CALLSITES, PersistentVector.EMPTY,
                 PROTOCOL_CALLSITES, PersistentVector.EMPTY,
                 VAR_CALLSITES, PersistentVector.EMPTY
              ));
      if(ret.isDeftype())
        {
        Var.pushThreadBindings(RT.map(METHOD, null,
                                      LOCAL_ENV, ret.fields
            , COMPILE_STUB_SYM, Symbol.intern(null, tagName)
            , COMPILE_STUB_CLASS, stub));
        }

      //now (methodname [args] body)*
      ret.line = (Integer) LINE.deref();
      IPersistentCollection methods = null;
      for(ISeq s = methodForms; s != null; s = RT.next(s))
        {
        NewInstanceMethod m = NewInstanceMethod.parse(ret, (ISeq) RT.first(s),thistag, overrideables);
        methods = RT.conj(methods, m);
        }


      ret.methods = methods;
      ret.keywords = (IPersistentMap) KEYWORDS.deref();
      ret.vars = (IPersistentMap) VARS.deref();
      ret.constants = (PersistentVector) CONSTANTS.deref();
      ret.constantsID = RT.nextID();
      ret.keywordCallsites = (IPersistentVector) KEYWORD_CALLSITES.deref();
      ret.protocolCallsites = (IPersistentVector) PROTOCOL_CALLSITES.deref();
      ret.varCallsites = (IPersistentVector) VAR_CALLSITES.deref();
      }
    finally
      {
      if(ret.isDeftype())
        Var.popThreadBindings();
      Var.popThreadBindings();
      }

    ret.compile(slashname(superClass),inames,false);
    ret.getCompiledClass();
    return ret;
    }

  /***
   * Current host interop uses reflection, which requires pre-existing classes
   * Work around this by:
   * Generate a stub class that has the same interfaces and fields as the class we are generating.
   * Use it as a type hint for this, and bind the simple name of the class to this stub (in resolve etc)
   * Unmunge the name (using a magic prefix) on any code gen for classes
   */
  static Class compileStub(String superName, NewInstanceExpr ret, String[] interfaceNames, Object frm){
    ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS);
    ClassVisitor cv = cw;
    cv.visit(V1_5, ACC_PUBLIC + ACC_SUPER, COMPILE_STUB_PREFIX + "/" + ret.internalName,
             null,superName,interfaceNames);

    //instance fields for closed-overs
    for(ISeq s = RT.keys(ret.closes); s != null; s = s.next())
      {
      LocalBinding lb = (LocalBinding) s.first();
      int access = ACC_PUBLIC + (ret.isVolatile(lb) ? ACC_VOLATILE :
                                 ret.isMutable(lb) ? 0 :
                                 ACC_FINAL);
      if(lb.getPrimitiveType() != null)
        cv.visitField(access
            , lb.name, Type.getType(lb.getPrimitiveType()).getDescriptor(),
                null, null);
      else
      //todo - when closed-overs are fields, use more specific types here and in ctor and emitLocal?
        cv.visitField(access
            , lb.name, OBJECT_TYPE.getDescriptor(), null, null);
      }

    //ctor that takes closed-overs and does nothing
    Method m = new Method("<init>", Type.VOID_TYPE, ret.ctorTypes());
    GeneratorAdapter ctorgen = new GeneratorAdapter(ACC_PUBLIC,
                                                    m,
                                                    null,
                                                    null,
                                                    cv);
    ctorgen.visitCode();
    ctorgen.loadThis();
    ctorgen.invokeConstructor(Type.getObjectType(superName), voidctor);
    ctorgen.returnValue();
    ctorgen.endMethod();

    if(ret.altCtorDrops > 0)
      {
      Type[] ctorTypes = ret.ctorTypes();
      Type[] altCtorTypes = new Type[ctorTypes.length-ret.altCtorDrops];
      for(int i=0;i<altCtorTypes.length;i++)
        altCtorTypes[i] = ctorTypes[i];
      Method alt = new Method("<init>", Type.VOID_TYPE, altCtorTypes);
      ctorgen = new GeneratorAdapter(ACC_PUBLIC,
                              alt,
                              null,
                              null,
                              cv);
      ctorgen.visitCode();
      ctorgen.loadThis();
      ctorgen.loadArgs();
      for(int i=0;i<ret.altCtorDrops;i++)
        ctorgen.visitInsn(Opcodes.ACONST_NULL);

      ctorgen.invokeConstructor(Type.getObjectType(COMPILE_STUB_PREFIX + "/" + ret.internalName),
                                new Method("<init>", Type.VOID_TYPE, ctorTypes));

      ctorgen.returnValue();
      ctorgen.endMethod();
      }
    //end of class
    cv.visitEnd();

    byte[] bytecode = cw.toByteArray();
    DynamicClassLoader loader = (DynamicClassLoader) LOADER.deref();
    return loader.defineClass(COMPILE_STUB_PREFIX + "." + ret.name, bytecode, frm);
  }

  static String[] interfaceNames(IPersistentVector interfaces){
    int icnt = interfaces.count();
    String[] inames = icnt > 0 ? new String[icnt] : null;
    for(int i=0;i<icnt;i++)
      inames[i] = slashname((Class) interfaces.nth(i));
    return inames;
  }


  static String slashname(Class c){
    return c.getName().replace('.', '/');
  }


  protected void emitMethods(ClassVisitor cv){
    for(ISeq s = RT.seq(methods); s != null; s = s.next())
      {
      ObjMethod method = (ObjMethod) s.first();
      method.emit(this, cv);
      }
    //emit bridge methods
    for(Map.Entry<IPersistentVector,Set<Class>> e : covariants.entrySet())
      {
      java.lang.reflect.Method m = mmap.get(e.getKey());
      Class[] params = m.getParameterTypes();
      Type[] argTypes = new Type[params.length];

      for(int i = 0; i < params.length; i++)
        {
        argTypes[i] = Type.getType(params[i]);
        }

      Method target = new Method(m.getName(), Type.getType(m.getReturnType()), argTypes);

      for(Class retType : e.getValue())
        {
             Method meth = new Method(m.getName(), Type.getType(retType), argTypes);

        GeneratorAdapter gen = new GeneratorAdapter(ACC_PUBLIC + ACC_BRIDGE,
                                                meth,
                                                null,
                                                //todo don't hardwire this
                                                EXCEPTION_TYPES,
                                                cv);
        gen.visitCode();
        gen.loadThis();
        gen.loadArgs();
        gen.invokeInterface(Type.getType(m.getDeclaringClass()),target);
        gen.returnValue();
        gen.endMethod();
        }
      }
  }

  static public IPersistentVector msig(java.lang.reflect.Method m){
    return RT.vector(m.getName(), RT.seq(m.getParameterTypes()),m.getReturnType());
  }

  static void considerMethod(java.lang.reflect.Method m, Map mm){
    IPersistentVector mk = msig(m);
    int mods = m.getModifiers();

    if(!(mm.containsKey(mk)
        || !(Modifier.isPublic(mods) || Modifier.isProtected(mods))
        || Modifier.isStatic(mods)
        || Modifier.isFinal(mods)))
      {
        mm.put(mk, m);
      }
  }

  static void gatherMethods(Class c, Map mm){
    for(; c != null; c = c.getSuperclass())
      {
      for(java.lang.reflect.Method m : c.getDeclaredMethods())
        considerMethod(m, mm);
      for(java.lang.reflect.Method m : c.getMethods())
        considerMethod(m, mm);
      }
  }

  static public Map[] gatherMethods(Class sc, ISeq interfaces){
    Map allm = new HashMap();
    gatherMethods(sc, allm);
    for(; interfaces != null; interfaces = interfaces.next())
      gatherMethods((Class) interfaces.first(), allm);

    Map<IPersistentVector,java.lang.reflect.Method> mm = new HashMap<IPersistentVector,java.lang.reflect.Method>();
    Map<IPersistentVector,Set<Class>> covariants = new HashMap<IPersistentVector,Set<Class>>();
    for(Object o : allm.entrySet())
      {
      Map.Entry e = (Map.Entry) o;
      IPersistentVector mk = (IPersistentVector) e.getKey();
      mk = (IPersistentVector) mk.pop();
      java.lang.reflect.Method m = (java.lang.reflect.Method) e.getValue();
      if(mm.containsKey(mk)) //covariant return
        {
        Set<Class> cvs = covariants.get(mk);
        if(cvs == null)
          {
          cvs = new HashSet<Class>();
          covariants.put(mk,cvs);
          }
        java.lang.reflect.Method om = mm.get(mk);
        if(om.getReturnType().isAssignableFrom(m.getReturnType()))
          {
          cvs.add(om.getReturnType());
          mm.put(mk, m);
          }
        else
          cvs.add(m.getReturnType());
        }
      else
        mm.put(mk, m);
      }
    return new Map[]{mm,covariants};
  }
}

public static class NewInstanceMethod extends ObjMethod{
  String name;
  Type[] argTypes;
  Type retType;
  Class retClass;
  Class[] exclasses;

  static Symbol dummyThis = Symbol.intern(null,"dummy_this_dlskjsdfower");
  private IPersistentVector parms;

  public NewInstanceMethod(ObjExpr objx, ObjMethod parent){
    super(objx, parent);
  }

  int numParams(){
    return argLocals.count();
  }

  String getMethodName(){
    return name;
  }

  Type getReturnType(){
    return retType;
  }

  Type[] getArgTypes(){
    return argTypes;
  }



  static public IPersistentVector msig(String name,Class[] paramTypes){
    return RT.vector(name,RT.seq(paramTypes));
  }

  static NewInstanceMethod parse(ObjExpr objx, ISeq form, Symbol thistag,
                                 Map overrideables) throws Exception{
    //(methodname [this-name args*] body...)
    //this-name might be nil
    NewInstanceMethod method = new NewInstanceMethod(objx, (ObjMethod) METHOD.deref());
    Symbol dotname = (Symbol)RT.first(form);
    Symbol name = (Symbol) Symbol.intern(null,munge(dotname.name)).withMeta(RT.meta(dotname));
    IPersistentVector parms = (IPersistentVector) RT.second(form);
    if(parms.count() == 0)
      {
      throw new IllegalArgumentException("Must supply at least one argument for 'this' in: " + dotname);
      }
    Symbol thisName = (Symbol) parms.nth(0);
    parms = RT.subvec(parms,1,parms.count());
    ISeq body = RT.next(RT.next(form));
    try
      {
      method.line = (Integer) LINE.deref();
      //register as the current method and set up a new env frame
            PathNode pnode =  new PathNode(PATHTYPE.PATH, (PathNode) CLEAR_PATH.get());
      Var.pushThreadBindings(
          RT.map(
              METHOD, method,
              LOCAL_ENV, LOCAL_ENV.deref(),
              LOOP_LOCALS, null,
              NEXT_LOCAL_NUM, 0
                            ,CLEAR_PATH, pnode
                            ,CLEAR_ROOT, pnode
                            ,CLEAR_SITES, PersistentHashMap.EMPTY
                    ));

      //register 'this' as local 0
      if(thisName != null)
        registerLocal((thisName == null) ? dummyThis:thisName,thistag, null,false);
      else
        getAndIncLocalNum();

      PersistentVector argLocals = PersistentVector.EMPTY;
      method.retClass = tagClass(tagOf(name));
      method.argTypes = new Type[parms.count()];
      boolean hinted = tagOf(name) != null;
      Class[] pclasses = new Class[parms.count()];
      Symbol[] psyms = new Symbol[parms.count()];

      for(int i = 0; i < parms.count(); i++)
        {
        if(!(parms.nth(i) instanceof Symbol))
          throw new IllegalArgumentException("params must be Symbols");
        Symbol p = (Symbol) parms.nth(i);
        Object tag = tagOf(p);
        if(tag != null)
          hinted = true;
        if(p.getNamespace() != null)
          p = Symbol.create(p.name);
        Class pclass = tagClass(tag);
        pclasses[i] = pclass;
        psyms[i] = p;
        }
      Map matches = findMethodsWithNameAndArity(name.name, parms.count(), overrideables);
      Object mk = msig(name.name, pclasses);
      java.lang.reflect.Method m = null;
      if(matches.size() > 0)
        {
        //multiple methods
        if(matches.size() > 1)
          {
          //must be hinted and match one method
          if(!hinted)
            throw new IllegalArgumentException("Must hint overloaded method: " + name.name);
          m = (java.lang.reflect.Method) matches.get(mk);
          if(m == null)
            throw new IllegalArgumentException("Can't find matching overloaded method: " + name.name);
          if(m.getReturnType() != method.retClass)
            throw new IllegalArgumentException("Mismatched return type: " + name.name +
            ", expected: " + m.getReturnType().getName()  + ", had: " + method.retClass.getName());
          }
        else  //one match
          {
          //if hinted, validate match,
          if(hinted)
            {
            m = (java.lang.reflect.Method) matches.get(mk);
            if(m == null)
              throw new IllegalArgumentException("Can't find matching method: " + name.name +
                                                 ", leave off hints for auto match.");
            if(m.getReturnType() != method.retClass)
              throw new IllegalArgumentException("Mismatched return type: " + name.name +
              ", expected: " + m.getReturnType().getName()  + ", had: " + method.retClass.getName());
            }
          else //adopt found method sig
            {
            m = (java.lang.reflect.Method) matches.values().iterator().next();
            method.retClass = m.getReturnType();
            pclasses = m.getParameterTypes();
            }
          }
        }
//      else if(findMethodsWithName(name.name,allmethods).size()>0)
//        throw new IllegalArgumentException("Can't override/overload method: " + name.name);
      else
        throw new IllegalArgumentException("Can't define method not in interfaces: " + name.name);

      //else
        //validate unque name+arity among additional methods

      method.retType = Type.getType(method.retClass);
      method.exclasses = m.getExceptionTypes();

      for(int i = 0; i < parms.count(); i++)
        {
        LocalBinding lb = registerLocal(psyms[i], null, new MethodParamExpr(pclasses[i]),true);
        argLocals = argLocals.assocN(i,lb);
        method.argTypes[i] = Type.getType(pclasses[i]);
        }
      for(int i = 0; i < parms.count(); i++)
        {
        if(pclasses[i] == long.class || pclasses[i] == double.class)
          getAndIncLocalNum();
        }
      LOOP_LOCALS.set(argLocals);
      method.name = name.name;
      method.methodMeta = RT.meta(name);
      method.parms = parms;
      method.argLocals = argLocals;
      method.body = (new BodyExpr.Parser()).parse(C.RETURN, body);
      return method;
      }
    finally
      {
      Var.popThreadBindings();
      }
  }

  private static Map findMethodsWithNameAndArity(String name, int arity, Map mm){
    Map ret = new HashMap();
    for(Object o : mm.entrySet())
      {
      Map.Entry e = (Map.Entry) o;
      java.lang.reflect.Method m = (java.lang.reflect.Method) e.getValue();
      if(name.equals(m.getName()) && m.getParameterTypes().length == arity)
        ret.put(e.getKey(), e.getValue());
      }
    return ret;
  }

  private static Map findMethodsWithName(String name, Map mm){
    Map ret = new HashMap();
    for(Object o : mm.entrySet())
      {
      Map.Entry e = (Map.Entry) o;
      java.lang.reflect.Method m = (java.lang.reflect.Method) e.getValue();
      if(name.equals(m.getName()))
        ret.put(e.getKey(), e.getValue());
      }
    return ret;
  }

  public void emit(ObjExpr obj, ClassVisitor cv){
    Method m = new Method(getMethodName(), getReturnType(), getArgTypes());

    Type[] extypes = null;
    if(exclasses.length > 0)
      {
      extypes = new Type[exclasses.length];
      for(int i=0;i<exclasses.length;i++)
        extypes[i] = Type.getType(exclasses[i]);
      }
    GeneratorAdapter gen = new GeneratorAdapter(ACC_PUBLIC,
                                                m,
                                                null,
                                                extypes,
                                                cv);
    addAnnotation(gen,methodMeta);
    for(int i = 0; i < parms.count(); i++)
      {
      IPersistentMap meta = RT.meta(parms.nth(i));
      addParameterAnnotation(gen, meta, i);
      }
    gen.visitCode();
    Label loopLabel = gen.mark();
    gen.visitLineNumber(line, loopLabel);
    try
      {
      Var.pushThreadBindings(RT.map(LOOP_LABEL, loopLabel, METHOD, this));
      MaybePrimitiveExpr be = (MaybePrimitiveExpr) body;
      if(Util.isPrimitive(retClass) && be.canEmitPrimitive())
        {
        if(be.getJavaClass() == retClass)
          be.emitUnboxed(C.RETURN,obj,gen);
        //todo - support the standard widening conversions
        else
          throw new IllegalArgumentException("Mismatched primitive return, expected: "
                                             + retClass + ", had: " + be.getJavaClass());
        }
      else
        {
        body.emit(C.RETURN, obj, gen);
        if(retClass == void.class)
          {
          gen.pop();
          }
        else
          gen.unbox(retType);
        }

      Label end = gen.mark();
      gen.visitLocalVariable("this", obj.objtype.getDescriptor(), null, loopLabel, end, 0);
      for(ISeq lbs = argLocals.seq(); lbs != null; lbs = lbs.next())
        {
        LocalBinding lb = (LocalBinding) lbs.first();
        gen.visitLocalVariable(lb.name, argTypes[lb.idx-1].getDescriptor(), null, loopLabel, end, lb.idx);
        }
      }
    catch(Exception e)
      {
      throw new RuntimeException(e);
      }
    finally
      {
      Var.popThreadBindings();
      }

    gen.returnValue();
    //gen.visitMaxs(1, 1);
    gen.endMethod();
  }
}

  static Class primClass(Symbol sym){
    if(sym == null)
      return null;
    Class c = null;
    if(sym.name.equals("int"))
      c = int.class;
    else if(sym.name.equals("long"))
      c = long.class;
    else if(sym.name.equals("float"))
      c = float.class;
    else if(sym.name.equals("double"))
      c = double.class;
    else if(sym.name.equals("char"))
      c = char.class;
    else if(sym.name.equals("short"))
      c = short.class;
    else if(sym.name.equals("byte"))
      c = byte.class;
    else if(sym.name.equals("boolean"))
      c = boolean.class;
    else if(sym.name.equals("void"))
      c = void.class;
    return c;
  }

  static Class tagClass(Object tag) throws Exception{
    if(tag == null)
      return Object.class;
    Class c = null;
    if(tag instanceof Symbol)
      c = primClass((Symbol) tag);
    if(c == null)
      c = HostExpr.tagToClass(tag);
    return c;
  }

static public class MethodParamExpr implements Expr, MaybePrimitiveExpr{
  final Class c;

  public MethodParamExpr(Class c){
    this.c = c;
  }

  public Object eval() throws Exception{
    throw new Exception("Can't eval");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    throw new RuntimeException("Can't emit");
  }

  public boolean hasJavaClass() throws Exception{
    return c != null;
  }

  public Class getJavaClass() throws Exception{
    return c;
  }

  public boolean canEmitPrimitive(){
    return Util.isPrimitive(c);
  }

  public void emitUnboxed(C context, ObjExpr objx, GeneratorAdapter gen){
    throw new RuntimeException("Can't emit");
  }
}

public static class CaseExpr extends UntypedExpr{
  public final LocalBindingExpr expr;
  public final int shift, mask, low, high;
  public final Expr defaultExpr;
  public final HashMap<Integer,Expr> tests;
  public final HashMap<Integer,Expr> thens;
  public final boolean allKeywords;

  public final int line;

  final static Method hashMethod = Method.getMethod("int hash(Object)");
  final static Method hashCodeMethod = Method.getMethod("int hashCode()");
  final static Method equalsMethod = Method.getMethod("boolean equals(Object, Object)");


  public CaseExpr(int line, LocalBindingExpr expr, int shift, int mask, int low, int high, Expr defaultExpr,
                  HashMap<Integer,Expr> tests,HashMap<Integer,Expr> thens, boolean allKeywords){
    this.expr = expr;
    this.shift = shift;
    this.mask = mask;
    this.low = low;
    this.high = high;
    this.defaultExpr = defaultExpr;
    this.tests = tests;
    this.thens = thens;
    this.line = line;
    this.allKeywords = allKeywords;
  }

  public Object eval() throws Exception{
    throw new UnsupportedOperationException("Can't eval case");
  }

  public void emit(C context, ObjExpr objx, GeneratorAdapter gen){
    Label defaultLabel = gen.newLabel();
    Label endLabel = gen.newLabel();
    HashMap<Integer,Label> labels = new HashMap();

    for(Integer i : tests.keySet())
      {
      labels.put(i, gen.newLabel());
      }

    Label[] la = new Label[(high-low)+1];

    for(int i=low;i<=high;i++)
      {
      la[i-low] = labels.containsKey(i) ? labels.get(i) : defaultLabel;
      }

    gen.visitLineNumber(line, gen.mark());
    expr.emit(C.EXPRESSION, objx, gen);
      gen.invokeStatic(UTIL_TYPE,hashMethod);
    gen.push(shift);
    gen.visitInsn(ISHR);
    gen.push(mask);
    gen.visitInsn(IAND);
    gen.visitTableSwitchInsn(low, high, defaultLabel, la);

    for(Integer i : labels.keySet())
      {
      gen.mark(labels.get(i));
      expr.emit(C.EXPRESSION, objx, gen);
      tests.get(i).emit(C.EXPRESSION, objx, gen);
      if(allKeywords)
        {
        gen.visitJumpInsn(IF_ACMPNE, defaultLabel);
        }
      else
        {
        gen.invokeStatic(UTIL_TYPE, equalsMethod);
        gen.ifZCmp(GeneratorAdapter.EQ, defaultLabel);
        }
      thens.get(i).emit(C.EXPRESSION,objx,gen);
      gen.goTo(endLabel);
      }

    gen.mark(defaultLabel);
    defaultExpr.emit(C.EXPRESSION, objx, gen);
    gen.mark(endLabel);
    if(context == C.STATEMENT)
      gen.pop();
  }

  static class Parser implements IParser{
    //(case* expr shift mask low high default map<minhash, [test then]> identity?)
    //prepared by case macro and presumed correct
    //case macro binds actual expr in let so expr is always a local,
    //no need to worry about multiple evaluation
    public Expr parse(C context, Object frm) throws Exception{
      ISeq form = (ISeq) frm;
      if(context == C.EVAL)
        return analyze(context, RT.list(RT.list(FN, PersistentVector.EMPTY, form)));
      PersistentVector args = PersistentVector.create(form.next());
      HashMap<Integer,Expr> tests = new HashMap();
      HashMap<Integer,Expr> thens = new HashMap();

            LocalBindingExpr testexpr = (LocalBindingExpr) analyze(C.EXPRESSION, args.nth(0));
      testexpr.shouldClear = false;
           
            PathNode branch = new PathNode(PATHTYPE.BRANCH, (PathNode) CLEAR_PATH.get());
      for(Object o : ((Map)args.nth(6)).entrySet())
        {
        Map.Entry e = (Map.Entry) o;
        Integer minhash = (Integer) e.getKey();
        MapEntry me = (MapEntry) e.getValue();
        Expr testExpr = new ConstantExpr(me.getKey());
        tests.put(minhash, testExpr);
                Expr thenExpr;
                try {
                    Var.pushThreadBindings(
                            RT.map(CLEAR_PATH, new PathNode(PATHTYPE.PATH,branch)));
                    thenExpr = analyze(context, me.getValue());
                    }
                finally{
                    Var.popThreadBindings();
                    }
        thens.put(minhash, thenExpr);
        }
           
            Expr defaultExpr;
            try {
                Var.pushThreadBindings(
                        RT.map(CLEAR_PATH, new PathNode(PATHTYPE.PATH,branch)));
                defaultExpr = analyze(context, args.nth(5));
                }
            finally{
                Var.popThreadBindings();
                }

      return new CaseExpr((Integer) LINE.deref(),
                        testexpr,
                        (Integer)args.nth(1),
                        (Integer)args.nth(2),
                        (Integer)args.nth(3),
                        (Integer)args.nth(4),
                        defaultExpr,
                        tests,thens,args.nth(7) != RT.F);

    }
  }
}

}
TOP

Related Classes of clojure.lang.Compiler$MethodParamExpr

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.
es of org.apache.ivy.core.module.descriptor.DefaultExcludeRule">org.apache.ivy.core.module.descriptor.DefaultExcludeRule
  • org.apache.ivy.core.module.descriptor.DefaultIncludeRule
  • 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.