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);
}
}
}