view src/jdk/nashorn/internal/ir/CallNode.java @ 143:4be452026847

8010652: Eliminate non-child references in Block/FunctionNode, and make few node types immutable Reviewed-by: jlaskey, lagergren
author attila
date Sat, 23 Mar 2013 00:58:39 +0100
parents e478708faa22
children
line wrap: on
line source

/*
 * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package jdk.nashorn.internal.ir;

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import jdk.nashorn.internal.codegen.types.Type;
import jdk.nashorn.internal.ir.annotations.Ignore;
import jdk.nashorn.internal.ir.visitor.NodeVisitor;
import jdk.nashorn.internal.runtime.Source;

/**
 * IR representation for a function call.
 *
 */
public class CallNode extends Node implements TypeOverride<CallNode> {

    private Type type;

    /** Function identifier or function body. */
    private Node function;

    /** Call arguments. */
    private List<Node> args;

    /** flag - is new expression */
    private boolean isNew;

    /** flag - is in with block */
    private boolean inWithBlock;

    /**
     * Arguments to be passed to builtin {@code eval} function
     */
    public static class EvalArgs {
        /** evaluated code */
        private Node code;

        /** 'this' passed to evaluated code */
        private IdentNode evalThis;

        /** location string for the eval call */
        final private String location;

        /** is this call from a strict context? */
        final private boolean strictMode;

        /**
         * Constructor
         *
         * @param code       code to evaluate
         * @param evalThis   this node
         * @param location   location for the eval call
         * @param strictMode is this a call from a strict context?
         */
        public EvalArgs(final Node code, final IdentNode evalThis, final String location, final boolean strictMode) {
            this.code = code;
            this.evalThis = evalThis;
            this.location = location;
            this.strictMode = strictMode;
        }

        /**
         * Return the code that is to be eval:ed by this eval function
         * @return code as an AST node
         */
        public Node getCode() {
            return code;
        }

        /**
         * Set the code that is to be eval.ed by this eval function
         * @param code the code as an AST node
         */
        public void setCode(final Node code) {
            this.code = code;
        }

        /**
         * Get the {@code this} symbol used to invoke this eval call
         * @return the {@code this} symbol
         */
        public IdentNode getThis() {
            return this.evalThis;
        }

        /**
         * Set the {@code this} symbol used to invoke this eval call
         * @param evalThis the {@code this} symbol
         */
        public void setThis(final IdentNode evalThis) {
            this.evalThis = evalThis;
        }

        /**
         * Get the human readable location for this eval call
         * @return the location
         */
        public String getLocation() {
            return this.location;
        }

        /**
         * Check whether this eval call is executed in strict mode
         * @return true if executed in strict mode, false otherwise
         */
        public boolean getStrictMode() {
            return this.strictMode;
        }
    }

    /** arguments for 'eval' call. Non-null only if this call node is 'eval' */
    @Ignore
    private EvalArgs evalArgs;

    /**
     * Constructors
     *
     * @param source   the source
     * @param token    token
     * @param finish   finish
     * @param function the function to call
     * @param args     args to the call
     */
    public CallNode(final Source source, final long token, final int finish, final Node function, final List<Node> args) {
        super(source, token, finish);

        setStart(function.getStart());

        this.function     = function;
        this.args         = args;
    }

    private CallNode(final CallNode callNode, final CopyState cs) {
        super(callNode);

        final List<Node> newArgs = new ArrayList<>();

        for (final Node arg : callNode.args) {
            newArgs.add(cs.existingOrCopy(arg));
        }

        this.function     = cs.existingOrCopy(callNode.function);     //TODO existing or same?
        this.args         = newArgs;
        this.isNew        = callNode.isNew;
        this.inWithBlock  = callNode.inWithBlock;
    }


    @Override
    public Type getType() {
        if (hasCallSiteType()) {
            return type;
        }
        return function instanceof FunctionNode ? ((FunctionNode)function).getReturnType() : Type.OBJECT;
    }

    @Override
    public CallNode setType(final Type type) {
        this.type = type;
        return this;
    }

    private boolean hasCallSiteType() {
        return this.type != null;
    }

    @Override
    public boolean canHaveCallSiteType() {
        return true;
    }

    @Override
    protected Node copy(final CopyState cs) {
        return new CallNode(this, cs);
    }

    /**
     * Assist in IR navigation.
     *
     * @param visitor IR navigating visitor.
     *
     * @return node or replacement
     */
    @Override
    public Node accept(final NodeVisitor visitor) {
        if (visitor.enterCallNode(this) != null) {
            function = function.accept(visitor);

            for (int i = 0, count = args.size(); i < count; i++) {
                args.set(i, args.get(i).accept(visitor));
            }

            return visitor.leaveCallNode(this);
        }

        return this;
    }

    @Override
    public void toString(final StringBuilder sb) {
        if (hasCallSiteType()) {
            sb.append('{');
            final String desc = getType().getDescriptor();
            sb.append(desc.charAt(desc.length() - 1) == ';' ? "O" : getType().getDescriptor());
            sb.append('}');
        }

        function.toString(sb);

        sb.append('(');

        boolean first = true;

        for (final Node arg : args) {
            if (!first) {
                sb.append(", ");
            } else {
                first = false;
            }

            arg.toString(sb);
        }

        sb.append(')');
    }

    /**
     * Get the arguments for the call
     * @return a list of arguments
     */
    public List<Node> getArgs() {
        return Collections.unmodifiableList(args);
    }

    /**
     * Reset the arguments for the call
     * @param args new arguments list
     */
    public void setArgs(final List<Node> args) {
        this.args = args;
    }

    /**
     * If this call is an {@code eval} call, get its EvalArgs structure
     * @return EvalArgs for call
     */
    public EvalArgs getEvalArgs() {
        return evalArgs;
    }

    /**
     * Set the EvalArgs structure for this call, if it has been determined it is an
     * {@code eval}
     *
     * @param evalArgs eval args
     */
    public void setEvalArgs(final EvalArgs evalArgs) {
        this.evalArgs = evalArgs;
    }

    /**
     * Check if this call is a call to {@code eval}
     * @return true if this is a call to {@code eval}
     */
    public boolean isEval() {
        return evalArgs != null;
    }

    /**
     * Return the function expression that this call invokes
     * @return the function
     */
    public Node getFunction() {
        return function;
    }

    /**
     * Reset the function expression that this call invokes
     * @param node the function
     */
    public void setFunction(final Node node) {
        function = node;
    }

    /**
     * Check if this call is a new operation
     * @return true if this a new operation
     */
    public boolean isNew() {
        return isNew;
    }

    /**
     * Flag this call as a new operation
     */
    public void setIsNew() {
        this.isNew = true;
    }

    /**
     * Check if this call is inside a {@code with} block
     * @return true if the call is inside a {@code with} block
     */
    public boolean inWithBlock() {
        return inWithBlock;
    }

    /**
     * Flag this call to be inside a {@code with} block
     */
    public void setInWithBlock() {
        this.inWithBlock = true;
    }
}