org.jruby.parser.IRStaticScope Maven / Gradle / Ivy
package org.jruby.parser;
import org.jruby.ast.AssignableNode;
import org.jruby.ast.DAsgnNode;
import org.jruby.ast.DVarNode;
import org.jruby.ast.LocalAsgnNode;
import org.jruby.ast.LocalVarNode;
import org.jruby.ast.Node;
import org.jruby.ast.VCallNode;
import org.jruby.ir.IRScope;
import org.jruby.lexer.yacc.ISourcePosition;
public class IRStaticScope extends StaticScope {
private static final String[] NO_NAMES = new String[0];
boolean isBlock; // Is this a block scope?
boolean isEval; // Is this an eval scope?
boolean isBlockOrEval;
boolean isArgumentScope; // Is this block and argument scope of a define_method (for the purposes of zsuper).
IRScope irScope; // Method/Closure that this static scope corresponds to
protected IRStaticScope(StaticScope enclosingScope, boolean isBlock, boolean isEval) {
this(enclosingScope, NO_NAMES, isBlock, isEval);
}
protected IRStaticScope(StaticScope enclosingScope, String[] names, boolean isBlock, boolean isEval) {
super(enclosingScope, names);
this.isBlock = isBlock;
this.isEval = isEval;
this.isBlockOrEval = isBlock || isEval;
this.isArgumentScope = !isBlockOrEval;
this.irScope = null;
}
public StaticScope getLocalScope() {
return (isEval || !isBlock) ? this : enclosingScope.getLocalScope();
}
public int isDefined(String name, int depth) {
if (isBlockOrEval) {
int slot = exists(name);
if (slot >= 0) return (depth << 16) | slot;
return enclosingScope.isDefined(name, depth + 1);
}
else {
return (depth << 16) | exists(name);
}
}
@Override
public boolean isBlockScope() {
return isBlockOrEval;
}
@Override
public boolean isArgumentScope() {
return isArgumentScope;
}
@Override
public void makeArgumentScope() {
this.isArgumentScope = true;
}
/**
* @see org.jruby.parser.StaticScope#getAllNamesInScope()
*/
public String[] getAllNamesInScope() {
String[] names = getVariables();
if (isBlockOrEval) {
String[] ourVariables = names;
String[] variables = enclosingScope.getAllNamesInScope();
// we know variables cannot be null since this IRStaticScope always returns a non-null array
names = new String[variables.length + ourVariables.length];
System.arraycopy(variables, 0, names, 0, variables.length);
System.arraycopy(ourVariables, 0, names, variables.length, ourVariables.length);
}
return names;
}
public AssignableNode addAssign(ISourcePosition position, String name, Node value) {
int slot = addVariable(name);
// No bit math to store level since we know level is zero for this case
return new DAsgnNode(position, name, slot, value);
}
public AssignableNode assign(ISourcePosition position, String name, Node value,
StaticScope topScope, int depth) {
int slot = exists(name);
// We can assign if we already have variable of that name here or we are the only
// scope in the chain (which Local scopes always are).
if (slot >= 0) {
return isBlockOrEval ? new DAsgnNode(position, name, ((depth << 16) | slot), value)
: new LocalAsgnNode(position, name, ((depth << 16) | slot), value);
} else if (!isBlockOrEval && (topScope == this)) {
slot = addVariable(name);
return new LocalAsgnNode(position, name, slot , value);
}
// If we are not a block-scope and we go there, we know that 'topScope' is a block scope
// because a local scope cannot be within a local scope
// If topScope was itself it would have created a LocalAsgnNode above.
return isBlockOrEval ? enclosingScope.assign(position, name, value, topScope, depth + 1)
: ((IRStaticScope)topScope).addAssign(position, name, value);
}
public Node declare(ISourcePosition position, String name, int depth) {
int slot = exists(name);
if (slot >= 0) {
return isBlockOrEval ? new DVarNode(position, ((depth << 16) | slot), name) : new LocalVarNode(position, ((depth << 16) | slot), name);
}
return isBlockOrEval ? enclosingScope.declare(position, name, depth + 1) : new VCallNode(position, name);
}
@Override
public String toString() {
return "IRStaticScope" + (isBlockOrEval ? "(BLOCK): " : "(LOCAL): ") + super.toString();
}
public IRScope getIRScope() {
return irScope;
}
public void setIRScope(IRScope irScope) {
this.irScope = irScope;
}
@Override
public int getNumberOfVariables() {
return (irScope == null) ? super.getNumberOfVariables() : irScope.getUsedVariablesCount();
}
@Override
public Type getType() {
return isBlock ? Type.BLOCK :
isEval ? Type.EVAL :
Type.LOCAL;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy