hacks to cope with org.ibex.xt
[org.ibex.js.git] / src / org / ibex / js / JSFunction.java
1 // Copyright 2004 Adam Megacz, see the COPYING file for licensing [GPL]
2 package org.ibex.js;
3
4 import java.io.*;
5 import org.ibex.util.*;
6
7 /** A JavaScript function, compiled into bytecode */
8 class JSFunction extends JS implements ByteCodes, Tokens, Task {
9
10
11     // Fields and Accessors ///////////////////////////////////////////////
12
13     int numFormalArgs = 0;         ///< the number of formal arguments
14
15     String sourceName;             ///< the source code file that this block was drawn from
16     private int firstLine = -1;    ///< the first line of this script
17
18     int[] line = new int[10];      ///< the line numbers
19     int[] op = new int[10];        ///< the instructions
20     Object[] arg = new Object[10]; ///< the arguments to the instructions
21     int size = 0;                  ///< the number of instruction/argument pairs
22
23     JSScope parentScope;           ///< the default scope to use as a parent scope when executing this
24     Vec formalArgs = new Vec();
25
26     // Public //////////////////////////////////////////////////////////////////////////////
27
28     // FEATURE: make sure that this can only be called from the Scheduler...
29     /** if you enqueue a function, it gets invoked in its own pauseable context */
30     public void perform() throws JSExn {
31         Interpreter i = new Interpreter(this, true, new JSArray());
32         i.resume();
33     }
34
35     /** parse and compile a function */
36     public static JSFunction _fromReader(String sourceName, int firstLine, Reader sourceCode) throws IOException {
37         JSFunction ret = new JSFunction(sourceName, firstLine, null);
38         if (sourceCode == null) return ret;
39         Parser p = new Parser(sourceCode, sourceName, firstLine);
40         while(true) {
41             int s = ret.size;
42             p.parseStatement(ret, null);
43             if (s == ret.size) break;
44         }
45         ret.add(-1, LITERAL, null); 
46         ret.add(-1, RETURN);
47         return ret;
48     }
49
50     public JSFunction _cloneWithNewParentScope(JSScope s) {
51         JSFunction ret = new JSFunction(sourceName, firstLine, s);
52         // Reuse the same op, arg, line, and size variables for the new "instance" of the function
53         // NOTE: Neither *this* function nor the new function should be modified after this call
54         ret.op = this.op;
55         ret.arg = this.arg;
56         ret.line = this.line;
57         ret.size = this.size;
58         ret.formalArgs = this.formalArgs;
59         ret.numFormalArgs = this.numFormalArgs;
60         return ret;
61     }
62
63     /** Note: code gets run in an <i>unpauseable</i> context. */
64     public Object call(Object a0, Object a1, Object a2, Object[] rest, int nargs) throws JSExn {
65         JSArray args = new JSArray();
66         if (nargs > 0) args.addElement(a0);
67         if (nargs > 1) args.addElement(a1);
68         if (nargs > 2) args.addElement(a2);
69         for(int i=3; i<nargs; i++) args.addElement(rest[i-3]);
70         Interpreter cx = new Interpreter(this, false, args);
71         return cx.resume();
72     }
73
74     public JSScope getParentScope() { return parentScope; }
75
76     // Adding and Altering Bytecodes ///////////////////////////////////////////////////
77
78     JSFunction(String sourceName, int firstLine, JSScope parentScope) {
79         this.sourceName = sourceName;
80         this.firstLine = firstLine;
81         this.parentScope = parentScope;
82     }
83
84     int get(int pos) { return op[pos]; }
85     Object getArg(int pos) { return arg[pos]; }
86     void set(int pos, int op_, Object arg_) { op[pos] = op_; arg[pos] = arg_; }
87     void set(int pos, Object arg_) { arg[pos] = arg_; }
88     int pop() { size--; arg[size] = null; return op[size]; }
89     void paste(JSFunction other) { for(int i=0; i<other.size; i++) add(other.line[i], other.op[i], other.arg[i]); }
90     JSFunction add(int line, int op_) { return add(line, op_, null); }
91     JSFunction add(int line, int op_, Object arg_) {
92         if (size == op.length - 1) {
93             int[] line2 = new int[op.length * 2]; System.arraycopy(this.line, 0, line2, 0, op.length); this.line = line2;
94             Object[] arg2 = new Object[op.length * 2]; System.arraycopy(arg, 0, arg2, 0, arg.length); arg = arg2;
95             int[] op2 = new int[op.length * 2]; System.arraycopy(op, 0, op2, 0, op.length); op = op2;
96         }
97         this.line[size] = line;
98         op[size] = op_;
99         arg[size] = arg_;
100         size++;
101         return this;
102     }
103     
104
105     // Debugging //////////////////////////////////////////////////////////////////////
106
107     public String toString() { return "JSFunction [" + sourceName + ":" + firstLine + "]"; }
108
109     public String dump() {
110         StringBuffer sb = new StringBuffer(1024);
111         sb.append("\n" + sourceName + ": " + firstLine + "\n");
112         for (int i=0; i < size; i++) {
113             sb.append(i).append(" (").append(line[i]).append(") :");
114             if (op[i] < 0) sb.append(bytecodeToString[-op[i]]);
115             else sb.append(codeToString[op[i]]);
116             sb.append(" ");
117             sb.append(arg[i] == null ? "(no arg)" : arg[i]);
118             if((op[i] == JF || op[i] == JT || op[i] == JMP) && arg[i] != null && arg[i] instanceof Number) {
119                 sb.append(" jump to ").append(i+((Number) arg[i]).intValue());
120             } else  if(op[i] == TRY) {
121                 int[] jmps = (int[]) arg[i];
122                 sb.append(" catch: ").append(jmps[0] < 0 ? "No catch block" : ""+(i+jmps[0]));
123                 sb.append(" finally: ").append(jmps[1] < 0 ? "No finally block" : ""+(i+jmps[1]));
124             }
125             sb.append("\n");
126         }
127         return sb.toString();
128     } 
129
130
131 }
132