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

com.oracle.truffle.api.Scope Maven / Gradle / Ivy

/*
 * Copyright (c) 2017, 2020, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * The Universal Permissive License (UPL), Version 1.0
 *
 * Subject to the condition set forth below, permission is hereby granted to any
 * person obtaining a copy of this software, associated documentation and/or
 * data (collectively the "Software"), free of charge and under any and all
 * copyright rights in the Software, and any and all patent rights owned or
 * freely licensable by each licensor hereunder covering either (i) the
 * unmodified Software as contributed to or provided by such licensor, or (ii)
 * the Larger Works (as defined below), to deal in both
 *
 * (a) the Software, and
 *
 * (b) any piece of software and/or hardware listed in the lrgrwrks.txt file if
 * one is included with the Software each a "Larger Work" to which the Software
 * is contributed by such licensors),
 *
 * without restriction, including without limitation the rights to copy, create
 * derivative works of, display, perform, and distribute the Software and make,
 * use, sell, offer for sale, import, export, have made, and have sold the
 * Software and the Larger Work(s), and to sublicense the foregoing rights on
 * either these or other terms.
 *
 * This license is subject to the following condition:
 *
 * The above copyright notice and either this complete permission notice or at a
 * minimum a reference to the UPL must be included in all copies or substantial
 * portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
 * SOFTWARE.
 */
package com.oracle.truffle.api;

import java.util.Objects;

import com.oracle.truffle.api.frame.Frame;
import com.oracle.truffle.api.nodes.Node;
import com.oracle.truffle.api.nodes.RootNode;

/**
 * Representation of a scope in a guest language program. The scope contains a set of declared and
 * valid variables. The scopes can be both lexical and dynamic.
 *
 * @since 0.30
 * @deprecated in 20.3, use {@link com.oracle.truffle.api.interop.NodeLibrary} and
 *             {@link com.oracle.truffle.api.interop.InteropLibrary#isScope(Object)} instead.
 */
@Deprecated
public final class Scope {

    private static final Scope EMPTY = new Scope();

    private final String name;
    private final Node node;
    private final Object arguments;
    private final Object variables;
    private final Object receiver;
    private final String receiverName;
    private final Object rootInstance;

    private Scope() {
        name = null;
        node = null;
        arguments = null;
        variables = null;
        receiver = null;
        receiverName = null;
        rootInstance = null;
    }

    Scope(String name, Node node, Object arguments, Object variables, Object receiver, String receiverName, Object rootInstance) {
        this.name = name;
        this.node = node;
        assertNullOrTruffleObject(arguments);
        this.arguments = arguments;
        assertTruffleObject(variables);
        this.variables = variables;
        assertNullOrInteropType(receiver);
        this.receiver = receiver;
        this.receiverName = receiverName;
        assertNullOrInteropType(rootInstance);
        this.rootInstance = rootInstance;
    }

    private static boolean assertTruffleObject(Object obj) {
        assert LanguageAccessor.interopAccess().isTruffleObject(obj) : Objects.toString(obj);
        return true;
    }

    private static void assertNullOrTruffleObject(Object obj) {
        assert obj == null || assertTruffleObject(obj);
    }

    private static void assertNullOrInteropType(Object obj) {
        if (obj != null) {
            LanguageAccessor.interopAccess().checkInteropType(obj);
        }
    }

    private static void assertNullOrExecutable(Object obj) {
        if (obj != null) {
            assert LanguageAccessor.interopAccess().isExecutableObject(obj) : Objects.toString(obj);
        }
    }

    /**
     * Create a new Scope builder.
     * 

* The properties representing the variables needs to have deterministic iteration order, * variable declaration order is recommended. * * @param name name of the scope, a name description like block, name of a function, closure, * script, module, etc. * @param variables a {@link com.oracle.truffle.api.interop.TruffleObject} containing the * variables as its properties, never null. Must respond true to * HAS_KEYS. * @since 0.30 */ public static Builder newBuilder(String name, Object variables) { return EMPTY.new Builder(name, variables); } /** * Human readable name of this scope. A name description like block, name of a function, * closure, script, module, etc. * * @since 0.30 */ public String getName() { return name; } /** * Get a node representing this scope. Functional scopes return the appropriate {@link RootNode} * , top scopes do not have a node associated. * * @return the node, or null when no node is associated. * @since 0.30 */ public Node getNode() { return node; } /** * Get variables declared in this scope. When a {@link Node} (and a {@link Frame}) was provided * when this scope was obtained, variables are valid at that specified {@link Node} (and * {@link Frame}). *

* The properties representing the variables have deterministic iteration order, variable * declaration order is preferred. * * @return A {@link com.oracle.truffle.api.interop.TruffleObject} containing the variables as * its properties, never null. * @since 0.30 */ public Object getVariables() { return variables; } /** * Get arguments of this scope. There might be different arguments returned in local scopes when * different {@link Frame} instances were provided. *

* The properties representing the arguments have deterministic iteration order, argument * declaration order is preferred. * * @return A {@link com.oracle.truffle.api.interop.TruffleObject} containing the arguments as * its properties for named arguments, or as its array for unnamed arguments. A * null is returned when this scope does not have a concept of arguments. * An empty TruffleObject is provided when it has a sense to have arguments (e.g. * function scope), but there are none. * @since 0.30 */ public Object getArguments() { return arguments; } /** * Get a receiver object of this scope. The receiver object is represented as this * in Java or JavaScript and self in Ruby, for instance. *

* The scope that represents the function provide receiver object, if there is one, other scopes * do not provide it, unless they override it. * * @return the receiver object, or null when there is no receiver * @since 19.0 * @see #getReceiverName() */ public Object getReceiver() { return receiver; } /** * Get the root instance object of this scope. The object is an instance of guest language * representation of the root node of this scope, e.g. a guest language function. * * @return the root instance object, or null when no such instance exists. * @since 19.3.0 */ public Object getRootInstance() { return rootInstance; } /** * Get code name of the receiver object, if there is any. * * @return the code name of {@link #getReceiver() receiver object}, or null. * @since 19.0 * @see #getReceiver() */ public String getReceiverName() { return receiverName; } /** * Builder to create a new {@link Scope} object. * * @since 0.30 * @deprecated in 20.3, use {@link com.oracle.truffle.api.interop.NodeLibrary} and * {@link com.oracle.truffle.api.interop.InteropLibrary#isScope(Object)} instead. */ @Deprecated public final class Builder { private final String name; private Node node; private Object arguments; private final Object variables; private Object receiver; private String receiverName; private Object rootInstance; Builder(String name, Object variables) { assert name != null; assertTruffleObject(variables); this.name = name; this.variables = variables; } /** * Set node representing the scope. It is expected that the appropriate * block/function/closure/etc. node is provided. * * @param node the scope's node * @since 0.30 */ @SuppressWarnings("hiding") public Builder node(Node node) { this.node = node; return this; } /** * Set arguments of the scope. *

* The properties representing the arguments needs to have deterministic iteration order, * argument declaration order is recommended. * * @param arguments arguments of the scope * @since 0.30 */ @SuppressWarnings("hiding") public Builder arguments(Object arguments) { assertNullOrTruffleObject(arguments); this.arguments = arguments; return this; } /** * Set a receiver object of this scope. The receiver object is represented as * this in Java or JavaScript and self in Ruby, for instance. *

* The scope that represents the function provide receiver object, if there is one, other * scopes do not provide it, unless they override it. * * @param name code name of the receiver object * @param receiver the receiver object * @since 19.0 */ @SuppressWarnings("hiding") public Builder receiver(String name, Object receiver) { assertNullOrInteropType(receiver); this.receiverName = name; this.receiver = receiver; return this; } /** * Set the root instance object of this scope. The object is an instance of guest language * representation of the root node of this scope, e.g. a guest language function. It's * supposed to be executable. * * @param rootInstance the root instance object, or null when no such instance * exists. * @since 19.3.0 */ @SuppressWarnings("hiding") public Builder rootInstance(Object rootInstance) { assertNullOrInteropType(rootInstance); assertNullOrExecutable(rootInstance); this.rootInstance = rootInstance; return this; } /** * Uses configuration of this builder to create new {@link Scope} object. * * @since 0.30 */ public Scope build() { return new Scope(name, node, arguments, variables, receiver, receiverName, rootInstance); } } }