All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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