Integration of VariableTable into ScriptOrFnNode to avoid the need to have a separated wrapper class around ObjArray/ObjToIntMap

This commit is contained in:
igor%mir2.org 2003-03-04 15:10:20 +00:00
parent 86a23ee950
commit c6efcf460c
10 changed files with 83 additions and 169 deletions

View File

@ -107,7 +107,7 @@ public class FunctionNode extends ScriptOrFnNode {
if (fn.getFunctionType() == FUNCTION_EXPRESSION_STATEMENT) {
String name = fn.getFunctionName();
if (name != null && name.length() != 0) {
removeParameterOrVar(name);
removeParamOrVar(name);
}
}
}
@ -119,7 +119,7 @@ public class FunctionNode extends ScriptOrFnNode {
Node stmts = getLastChild();
if (getFunctionType() == FUNCTION_EXPRESSION) {
String name = getFunctionName();
if (name != null && name.length() != 0 && !hasParameterOrVar(name))
if (name != null && name.length() != 0 && !hasParamOrVar(name))
{
// A function expression needs to have its name as a
// variable (if it isn't already allocated as a variable).

View File

@ -218,7 +218,7 @@ public class Interpreter {
itsData.itsDoubleTable = tmp;
}
itsData.itsMaxVars = scriptOrFn.getParameterAndVarCount();
itsData.itsMaxVars = scriptOrFn.getParamAndVarCount();
// itsMaxFrameArray: interpret method needs this amount for its
// stack and sDbl arrays
itsData.itsMaxFrameArray = itsData.itsMaxVars
@ -226,8 +226,8 @@ public class Interpreter {
+ itsData.itsMaxTryDepth
+ itsData.itsMaxStack;
itsData.argNames = scriptOrFn.getParameterAndVarNames();
itsData.argCount = scriptOrFn.getParameterCount();
itsData.argNames = scriptOrFn.getParamAndVarNames();
itsData.argCount = scriptOrFn.getParamCount();
}
private int updateLineNumber(Node node, int iCodeTop) {
@ -674,7 +674,7 @@ public class Interpreter {
// use typeofname if an activation frame exists
// since the vars all exist there instead of in jregs
if (itsInFunctionFlag && !itsData.itsNeedsActivation)
index = scriptOrFn.getParameterOrVarIndex(name);
index = scriptOrFn.getParamOrVarIndex(name);
if (index == -1) {
iCodeTop = addByte(TokenStream.TYPEOFNAME, iCodeTop);
iCodeTop = addString(name, iCodeTop);
@ -724,7 +724,7 @@ public class Interpreter {
iCodeTop);
itsStackDepth--;
} else {
int i = scriptOrFn.getParameterOrVarIndex(name);
int i = scriptOrFn.getParamOrVarIndex(name);
iCodeTop = addByte(type == TokenStream.INC
? TokenStream.VARINC
: TokenStream.VARDEC,
@ -930,7 +930,7 @@ public class Interpreter {
iCodeTop = addByte(TokenStream.GETPROP, iCodeTop);
itsStackDepth--;
} else {
int index = scriptOrFn.getParameterOrVarIndex(name);
int index = scriptOrFn.getParamOrVarIndex(name);
iCodeTop = addByte(TokenStream.GETVAR, iCodeTop);
iCodeTop = addByte(index, iCodeTop);
itsStackDepth++;
@ -949,7 +949,7 @@ public class Interpreter {
String name = child.getString();
child = child.getNext();
iCodeTop = generateICode(child, iCodeTop);
int index = scriptOrFn.getParameterOrVarIndex(name);
int index = scriptOrFn.getParamOrVarIndex(name);
iCodeTop = addByte(TokenStream.SETVAR, iCodeTop);
iCodeTop = addByte(index, iCodeTop);
}

View File

@ -305,7 +305,7 @@ public class JavaAdapter extends ScriptableObject {
ScriptableObject.getProperty(p, "length"));
} else if (f instanceof FunctionNode) {
// This is used only by optimizer/Codegen
length = ((FunctionNode)f).getParameterCount();
length = ((FunctionNode)f).getParamCount();
} else {
continue;
}

View File

@ -376,7 +376,7 @@ public class NodeTransformer {
// use of "arguments" requires an activation object.
((FunctionNode) tree).setRequiresActivation(true);
}
if (tree.hasParameterOrVar(name)) {
if (tree.hasParamOrVar(name)) {
if (type == TokenStream.SETNAME) {
node.setType(TokenStream.SETVAR);
bind.setType(TokenStream.STRING);
@ -416,7 +416,7 @@ public class NodeTransformer {
// Use of "arguments" requires an activation object.
((FunctionNode) tree).setRequiresActivation(true);
}
if (tree.hasParameterOrVar(name)) {
if (tree.hasParamOrVar(name)) {
node.setType(TokenStream.GETVAR);
}
break;
@ -458,7 +458,7 @@ public class NodeTransformer {
boolean addGetThis = false;
if (left.getType() == TokenStream.NAME) {
String name = left.getString();
if (inFunction && tree.hasParameterOrVar(name)
if (inFunction && tree.hasParamOrVar(name)
&& !inWithStatement())
{
// call to a var. Transform to Call(GetVar("a"), b, c)

View File

@ -261,11 +261,11 @@ class Parser {
first = false;
mustMatchToken(ts, ts.NAME, "msg.no.parm");
String s = ts.getString();
if (fnNode.hasParameterOrVar(s)) {
if (fnNode.hasParamOrVar(s)) {
Object[] msgArgs = { s };
ts.reportCurrentLineWarning("msg.dup.parms", msgArgs);
}
fnNode.addParameter(s);
fnNode.addParam(s);
sourceAddString(ts.NAME, s);
} while (ts.matchToken(ts.COMMA));

View File

@ -117,48 +117,65 @@ public class ScriptOrFnNode extends Node {
return regexps.size() / 2 - 1;
}
public final boolean hasParameterOrVar(String name) {
if (variableTable == null) { return false; }
return variableTable.hasVariable(name);
public final boolean hasParamOrVar(String name) {
return itsVariableNames.has(name);
}
public final int getParameterOrVarIndex(String name) {
if (variableTable == null) { return -1; }
return variableTable.getOrdinal(name);
public final int getParamOrVarIndex(String name) {
return itsVariableNames.get(name, -1);
}
public final String getParameterOrVarName(int index) {
return variableTable.getVariable(index);
public final String getParamOrVarName(int index) {
return (String)itsVariables.get(index);
}
public final int getParameterCount() {
if (variableTable == null) { return 0; }
return variableTable.getParameterCount();
public final int getParamCount() {
return varStart;
}
public final int getParameterAndVarCount() {
if (variableTable == null) { return 0; }
return variableTable.size();
public final int getParamAndVarCount() {
return itsVariables.size();
}
public final String[] getParameterAndVarNames() {
if (variableTable == null) { return new String[0]; }
return variableTable.getAllVariables();
public final String[] getParamAndVarNames() {
String[] array = new String[itsVariables.size()];
itsVariables.toArray(array);
return array;
}
public final void addParameter(String name) {
if (variableTable == null) { variableTable = new VariableTable(); }
variableTable.addParameter(name);
public final void addParam(String name) {
// Check addparam is not called after addLocal
if (varStart != itsVariables.size()) Context.codeBug();
// Allow non-unique parameter names: use the last occurrence
int index = varStart++;
itsVariables.add(name);
itsVariableNames.put(name, index);
}
public final void addVar(String name) {
if (variableTable == null) { variableTable = new VariableTable(); }
variableTable.addLocal(name);
int vIndex = itsVariableNames.get(name, -1);
if (vIndex != -1) {
// There's already a variable or parameter with this name.
return;
}
int index = itsVariables.size();
itsVariables.add(name);
itsVariableNames.put(name, index);
}
public final void removeParameterOrVar(String name) {
if (variableTable == null) { return; }
variableTable.removeLocal(name);
public final void removeParamOrVar(String name) {
int i = itsVariableNames.get(name, -1);
if (i != -1) {
itsVariables.remove(i);
itsVariableNames.remove(name);
ObjToIntMap.Iterator iter = itsVariableNames.newIterator();
for (iter.start(); !iter.done(); iter.next()) {
int v = iter.getValue();
if (v > i) {
iter.setValue(v - 1);
}
}
}
}
public final int getLocalCount() { return localCount; }
@ -174,9 +191,19 @@ public class ScriptOrFnNode extends Node {
private String sourceName;
private int baseLineno = -1;
private int endLineno = -1;
private ObjArray functions;
private ObjArray regexps;
private VariableTable variableTable;
// a list of the formal parameters and local variables
private ObjArray itsVariables = new ObjArray();
// mapping from name to index in list
private ObjToIntMap itsVariableNames = new ObjToIntMap(11);
private int varStart; // index in list of first variable
private int localCount;
}

View File

@ -1,113 +0,0 @@
/* -*- Mode: java; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
*
* The contents of this file are subject to the Netscape Public
* License Version 1.1 (the "License"); you may not use this file
* except in compliance with the License. You may obtain a copy of
* the License at http://www.mozilla.org/NPL/
*
* Software distributed under the License is distributed on an "AS
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
* implied. See the License for the specific language governing
* rights and limitations under the License.
*
* The Original Code is Rhino code, released
* May 6, 1999.
*
* The Initial Developer of the Original Code is Netscape
* Communications Corporation. Portions created by Netscape are
* Copyright (C) 1997-1999 Netscape Communications Corporation. All
* Rights Reserved.
*
* Contributor(s):
* Roger Lawrence
* Igor Bukanov
*
* Alternatively, the contents of this file may be used under the
* terms of the GNU Public License (the "GPL"), in which case the
* provisions of the GPL are applicable instead of those above.
* If you wish to allow use of your version of this file only
* under the terms of the GPL and not to allow others to use your
* version of this file under the NPL, indicate your decision by
* deleting the provisions above and replace them with the notice
* and other provisions required by the GPL. If you do not delete
* the provisions above, a recipient may use your version of this
* file under either the NPL or the GPL.
*/
package org.mozilla.javascript;
import java.io.*;
public class VariableTable {
public int size() {
return itsVariables.size();
}
public int getParameterCount(){
return varStart;
}
public String getVariable(int index) {
return (String)itsVariables.get(index);
}
public boolean hasVariable(String name) {
return itsVariableNames.has(name);
}
public int getOrdinal(String name) {
return itsVariableNames.get(name, -1);
}
public String[] getAllVariables() {
String[] array = new String[itsVariables.size()];
itsVariables.toArray(array);
return array;
}
public void addParameter(String pName) {
// Check addParameter is not called after addLocal
if (varStart != itsVariables.size()) Context.codeBug();
// Allow non-unique parameter names: use the last occurrence
int index = varStart++;
itsVariables.add(pName);
itsVariableNames.put(pName, index);
}
public void addLocal(String vName) {
int vIndex = itsVariableNames.get(vName, -1);
if (vIndex != -1) {
// There's already a variable or parameter with this name.
return;
}
int index = itsVariables.size();
itsVariables.add(vName);
itsVariableNames.put(vName, index);
}
// This should only be called very early in compilation
public void removeLocal(String name) {
int i = itsVariableNames.get(name, -1);
if (i != -1) {
itsVariables.remove(i);
itsVariableNames.remove(name);
ObjToIntMap.Iterator iter = itsVariableNames.newIterator();
for (iter.start(); !iter.done(); iter.next()) {
int v = iter.getValue();
if (v > i) {
iter.setValue(v - 1);
}
}
}
}
// a list of the formal parameters and local variables
private ObjArray itsVariables = new ObjArray();
// mapping from name to index in list
private ObjToIntMap itsVariableNames = new ObjToIntMap(11);
private int varStart; // index in list of first variable
}

View File

@ -333,7 +333,7 @@ public class Codegen extends Interpreter {
+ "Ljava/lang/Object;",
flags);
int argCount = fnCurrent.getParameterCount();
int argCount = fnCurrent.getParamCount();
int firstLocal = (4 + argCount * 3) + 1;
aload((short)0); // this
@ -429,7 +429,7 @@ public class Codegen extends Interpreter {
addByteCode(ByteCode.ALOAD_1);
addByteCode(ByteCode.ALOAD_2);
addByteCode(ByteCode.ALOAD_3);
for (int i = 0; i < scriptOrFn.getParameterCount(); i++) {
for (int i = 0; i < scriptOrFn.getParamCount(); i++) {
push(i);
addByteCode(ByteCode.ALOAD, 4);
addByteCode(ByteCode.ARRAYLENGTH);
@ -465,7 +465,7 @@ public class Codegen extends Interpreter {
if (!fnCurrent.getParameterNumberContext()) {
// make sure that all parameters are objects
itsForcedObjectParameters = true;
for (int i = 0; i < fnCurrent.getParameterCount(); i++) {
for (int i = 0; i < fnCurrent.getParamCount(); i++) {
OptLocalVariable lVar = fnCurrent.getVar(i);
aload(lVar.getJRegister());
classFile.add(ByteCode.GETSTATIC,
@ -482,7 +482,7 @@ public class Codegen extends Interpreter {
markLabel(isObjectLabel);
}
}
generatePrologue(cx, true, scriptOrFn.getParameterCount());
generatePrologue(cx, true, scriptOrFn.getParamCount());
} else {
startNewMethod("call",
"(Lorg/mozilla/javascript/Context;" +
@ -544,7 +544,7 @@ public class Codegen extends Interpreter {
// 2 is reserved for parentScope
// 3 is reserved for script 'this'
short jReg = 4;
int parameterCount = fnCurrent.getParameterCount();
int parameterCount = fnCurrent.getParamCount();
for (int i = 0; i < parameterCount; i++) {
OptLocalVariable lVar = fnCurrent.getVar(i);
lVar.assignJRegister(jReg);
@ -1145,7 +1145,7 @@ public class Codegen extends Interpreter {
"functionName", "Ljava/lang/String;");
}
int N = scriptOrFn.getParameterAndVarCount();
int N = scriptOrFn.getParamAndVarCount();
if (N != 0) {
setNonTrivialInit(methodName);
push(N);
@ -1153,7 +1153,7 @@ public class Codegen extends Interpreter {
for (int i = 0; i != N; i++) {
addByteCode(ByteCode.DUP);
push(i);
push(scriptOrFn.getParameterOrVarName(i));
push(scriptOrFn.getParamOrVarName(i));
addByteCode(ByteCode.AASTORE);
}
addByteCode(ByteCode.ALOAD_0);
@ -1163,7 +1163,7 @@ public class Codegen extends Interpreter {
"argNames", "[Ljava/lang/String;");
}
int parmCount = scriptOrFn.getParameterCount();
int parmCount = scriptOrFn.getParamCount();
if (parmCount != 0) {
setNonTrivialInit(methodName);
addByteCode(ByteCode.ALOAD_0);
@ -1373,7 +1373,7 @@ public class Codegen extends Interpreter {
!((OptFunctionNode)scriptOrFn).requiresActivation();
if (hasVarsInRegs) {
// No need to create activation. Pad arguments if need be.
int parmCount = scriptOrFn.getParameterCount();
int parmCount = scriptOrFn.getParamCount();
if (inFunction && parmCount > 0 && directParameterCount < 0) {
// Set up args array
// check length of arguments, pad if need be

View File

@ -48,17 +48,17 @@ class OptFunctionNode extends FunctionNode {
protected void finishParsing(IRFactory irFactory) {
super.finishParsing(irFactory);
int N = getParameterAndVarCount();
int parameterCount = getParameterCount();
int N = getParamAndVarCount();
int parameterCount = getParamCount();
optVars = new OptLocalVariable[N];
for (int i = 0; i != N; ++i) {
String name = getParameterOrVarName(i);
String name = getParamOrVarName(i);
optVars[i] = new OptLocalVariable(name, i < parameterCount);
}
}
String getDirectCallParameterSignature() {
int pCount = getParameterCount();
int pCount = getParamCount();
switch (pCount) {
case 0: return ZERO_PARAM_SIG;
case 1: return ONE_PARAM_SIG;
@ -129,7 +129,7 @@ class OptFunctionNode extends FunctionNode {
}
OptLocalVariable getVar(String name) {
int index = getParameterOrVarIndex(name);
int index = getParamOrVarIndex(name);
if (index < 0) { return null; }
return optVars[index];
}

View File

@ -129,7 +129,7 @@ class OptTransformer extends NodeTransformer {
OptFunctionNode theFunction
= (OptFunctionNode)theFnClassNameList.get(targetName);
if (theFunction != null) {
int N = theFunction.getParameterCount();
int N = theFunction.getParamCount();
// Refuse to directCall any function with more
// than 32 parameters - prevent code explosion
// for wacky test cases