X-Git-Url: http://git.megacz.com/?a=blobdiff_plain;f=src%2Forg%2Fxwt%2Fjs%2FJS.java;h=ba83b01dcc72b80256dd069d8683a8d751e6ca93;hb=4444c6057bb0cee5d2ae5a55b3045fd8eb790295;hp=c05388145a79f663296d9390a974554fb771c384;hpb=56387d062db4aca0510daa34579aa139570bac87;p=org.ibex.core.git diff --git a/src/org/xwt/js/JS.java b/src/org/xwt/js/JS.java index c053881..ba83b01 100644 --- a/src/org/xwt/js/JS.java +++ b/src/org/xwt/js/JS.java @@ -1,203 +1,270 @@ -// Copyright 2002 Adam Megacz, see the COPYING file for licensing [GPL] +// Copyright 2003 Adam Megacz, see the COPYING file for licensing [GPL] package org.xwt.js; import org.xwt.util.*; +import org.xwt.*; import java.io.*; import java.util.*; -/** The public API for the JS engine */ -// FEATURE: try using mutable, recycled 'Num' objects +/** + * The public API for the JS engine. JS itself is actually a class + * implementing the absolute minimal amount of functionality for an + * Object which can be manipulated by JavaScript code. The static + * methods, fields, and inner classes of JS define the publicly + * visible API for the XWT JavaScript engine; code outside this + * package should never depend on anything not defined in this file. + */ public abstract class JS { - // Static Methods ////////////////////////////////////////////////////////////////////// + // Public Helper Methods ////////////////////////////////////////////////////////////////////// - public static Function getCurrentFunction() { - return (Function)currentFunction.get(Thread.currentThread()); - } - public static String getCurrentFunctionSourceName() { - return getCurrentFunctionSourceName(Thread.currentThread()); - } - public static String getCurrentFunctionSourceName(Thread t) { - Function f = (Function)currentFunction.get(t); - if (f == null) return "null"; - return f.getSourceName(); - } - public static String getFileAndLine() { - return "unknown:??"; + /** parse and compile a function */ + public static Function parse(String sourceName, int firstLine, Reader sourceCode) throws IOException { + return new Function(sourceName, firstLine, sourceCode, null); } + /** coerce an object to a Boolean */ public static boolean toBoolean(Object o) { - if (o == null) return false; - if (o instanceof Boolean) return ((Boolean)o).booleanValue(); - if (o instanceof Number) return o.equals(new Integer(0)); - return true; + if (o == null) return false; + if (o instanceof Boolean) return ((Boolean)o).booleanValue(); + if (o instanceof Long) return ((Long)o).longValue() != 0; + if (o instanceof Integer) return ((Integer)o).intValue() != 0; + if (o instanceof Number) { + double d = ((Number) o).doubleValue(); + // NOTE: d == d is a test for NaN. It should be faster than Double.isNaN() + return d != 0.0 && d == d; + } + if (o instanceof String) return ((String)o).length() != 0; + return true; } + + /** coerce an object to a Long */ public static long toLong(Object o) { return toNumber(o).longValue(); } + + /** coerce an object to an Int */ + public static int toInt(Object o) { return toNumber(o).intValue(); } + + /** coerce an object to a Double */ public static double toDouble(Object o) { return toNumber(o).doubleValue(); } + + /** coerce an object to a Number */ public static Number toNumber(Object o) { - if (o == null) return new Long(0); - if (o instanceof Number) return ((Number)o); - if (o instanceof String) try { return new Double((String)o); } catch (NumberFormatException e) { return new Double(0); } - if (o instanceof Boolean) return ((Boolean)o).booleanValue() ? new Long(1) : new Long(0); - if (o instanceof JS) return ((JS)o).coerceToNumber(); - // FIXME - throw new Error("toNumber() got object of type " + o.getClass().getName()); - } + if (o == null) return new Long(0); + if (o instanceof Number) return ((Number)o); + // NOTE: There are about 3 pages of rules in ecma262 about string to number conversions + // We aren't even close to following all those rules. We probably never will be. + if (o instanceof String) + try { return new Double((String)o); } catch (NumberFormatException e) { return new Double(Double.NaN); } + if (o instanceof Boolean) return ((Boolean)o).booleanValue() ? new Long(1) : new Long(0); + if (o instanceof JS) return ((JS)o).coerceToNumber(); + throw new Error("toNumber() got object of type " + o.getClass().getName() + " which we don't know how to handle"); + } + + /** coerce an object to a String */ + public static String toString(Object o) { + if(o == null) return "null"; + if(o instanceof String) return (String) o; + if(o instanceof Integer || o instanceof Long || o instanceof Boolean) return o.toString(); + if(o instanceof JS) return ((JS)o).coerceToString(); + if(o instanceof Double || o instanceof Float) { + double d = ((Number)o).doubleValue(); + if((int)d == d) return Integer.toString((int)d); + return o.toString(); + } + return o.toString(); + } + // Instance Methods //////////////////////////////////////////////////////////////////// public abstract Object get(Object key) throws JS.Exn; - public abstract void put(Object key, Object val) throws JS.Exn; + public abstract Object put(Object key, Object val) throws JS.Exn; public abstract Object[] keys(); - - public Number coerceToNumber() { throw new Error("you cannot coerce a " + this.getClass().getName() + " into a Number"); } - public String coerceToString() { throw new Error("you cannot coerce a " + this.getClass().getName() + " into a String"); } - public boolean coerceToBoolean() { throw new Error("you cannot coerce a " + this.getClass().getName() + " into a Boolean"); } + public Object callMethod(Object method, Array args, boolean checkOnly) throws JS.Exn { + if (checkOnly) return Boolean.FALSE; + Object o = get(method); + if (o instanceof JS.Callable) return ((JS.Callable)o).call(args); + else if (o == null) throw new JS.Exn("Attempted to call non-existent method: " + method); + else throw new JS.Exn("Attempted to call a non-function: " +method); + } + + public Number coerceToNumber() { throw new JS.Exn("tried to coerce a JavaScript object to a Number"); } + public String coerceToString() { throw new JS.Exn("tried to coerce a JavaScript object to a String"); } + public boolean coerceToBoolean() { throw new JS.Exn("tried to coerce a JavaScript object to a Boolean"); } + public String typeName() { return "object"; } - // Subclasses ///////////////////////////////////////////////////////////////////////// + // Inner Classes ///////////////////////////////////////////////////////////////////////// + /** A sensible implementation of the abstract methods in the JS class */ public static class Obj extends JS { - private Hash entries = new Hash(); - private boolean sealed = false; - public Obj() { this(false); } - public Obj(boolean sealed) { this.sealed = sealed; } - public void setSeal(boolean sealed) { this.sealed = sealed; } - public Object get(Object key) { return entries.get(key); } - public void put(Object key, Object val) { if (!sealed) entries.put(key, val); } - public Object[] keys() { return(entries.keys()); } + private Hash entries = null; + private boolean sealed = false; + public Obj() { this(false); } + public Obj(boolean sealed) { this.sealed = sealed; } + public void setSeal(boolean sealed) { this.sealed = sealed; } ///< a sealed object cannot have its properties modified + public Object put(Object key, Object val) { put2(key, null, val); return null; } + protected void put2(Object key, Object key2, Object val) { + if (sealed) return; + if (entries == null) entries = new Hash(); + entries.put(key, key2, val); } + public Object[] keys() { return entries == null ? new Object[0] : entries.keys(); } + public Object get(Object key) { return get(key, null); } + protected Object get(Object key, Object key2) { + if (entries == null) return null; + if(key2 == null && callMethod((String)key, null, true) == Boolean.TRUE) + return new Internal.CallableStub(this, key); + return entries.get(key, key2); + } } + /** An exception which can be thrown and caught by JavaScript code */ public static class Exn extends RuntimeException { - private Object js = null; - public Exn(Object js) { this.js = js; } - public String toString() { return "JS.Exn: " + js.toString(); } - public String getMessage() { return toString(); } - public Object getObject() { return js; } + private Object js = null; + public Exn(Object js) { this.js = js; } + public String toString() { return "JS.Exn: " + js; } + public String getMessage() { return toString(); } + public Object getObject() { return js; } } - public static class Array extends Obj { - private Vec vec = new Vec(); - public Array() { } - public Array(int size) { vec.setSize(size); } - private static int intVal(Object o) { - if (o instanceof Number) { - int intVal = ((Number)o).intValue(); - if (intVal == ((Number)o).doubleValue()) return intVal; - return Integer.MIN_VALUE; - } - if (!(o instanceof String)) return Integer.MIN_VALUE; - String s = (String)o; - for(int i=0; i '9') return Integer.MIN_VALUE; - return Integer.parseInt(s); - } - public Object get(Object key) throws JS.Exn { - // FIXME: HACK! - if (key.equals("cascade")) return org.xwt.Trap.cascadeFunction; - if (key.equals("trapee")) return org.xwt.Trap.currentTrapee(); - if (key.equals("length")) return new Long(vec.size()); - int i = intVal(key); - if (i == Integer.MIN_VALUE) return super.get(key); - try { - return vec.elementAt(i); - } catch (ArrayIndexOutOfBoundsException e) { - throw new JS.Exn(e.getMessage()); - } - } - public void put(Object key, Object val) { - if (key.equals("length")) vec.setSize(toNumber(val).intValue()); - int i = intVal(key); - if (i == Integer.MIN_VALUE) super.put(key, val); - else { - if (i >= vec.size()) vec.setSize(i+1); - vec.setElementAt(val, i); - } - } - public Object[] keys() { - Object[] sup = super.keys(); - Object[] ret = new Object[vec.size() + 1 + sup.length]; - System.arraycopy(sup, 0, ret, vec.size(), sup.length); - for(int i=0; i= f.size) ? -1 : f.line[pc]; } + public static int getLine() { return current().getLine_(); } + public static String getSourceName() { return current().f == null ? null : current().f.sourceName; } + + /** fetches the currently-executing javascript function */ + public static JS.Context current() { return (JS.Context)threadToContext.get(Thread.currentThread()); } + private static Hashtable threadToContext = new Hashtable(); + + // Instance members and methods ////////////////////////////////////////////////////////////////////// + + /** the currently-executing Function */ + Function f = null; + + /** the currently-executing scope */ + public Scope scope = null; // FIXME: do we really need this? the function should contain this info + + /** the object stack */ + Vec stack = new Vec(); + + /** the program counter */ + int pc = 0; + + public Context(Function function, Scope scope) { + if (scope == null) scope = new Scope(function.parentScope); + stack.push(new Function.CallMarker(this)); + stack.push(new JS.Array()); + this.f = function; + this.scope = scope; + } + + public Object resume() { return resume(null); } + public Object resume(Object o) { + Thread t = Thread.currentThread(); + Context old = (Context)threadToContext.get(t); + try { + threadToContext.put(t, this); + stack.push(o); + return Function.eval(this); + } finally { + if (old == null) threadToContext.remove(t); + else threadToContext.put(t, old); + } + } + + } }