001/* 002 * Copyright (c) 2012-2021 Institut National des Sciences Appliquées de Lyon (INSA Lyon) and others 003 * 004 * This program and the accompanying materials are made available under the 005 * terms of the Eclipse Public License 2.0 which is available at 006 * http://www.eclipse.org/legal/epl-2.0. 007 * 008 * SPDX-License-Identifier: EPL-2.0 009 */ 010 011package gololang.ir; 012 013import java.util.LinkedList; 014 015public abstract class ExpressionStatement<T extends ExpressionStatement<T>> extends GoloStatement<T> { 016 017 private final LinkedList<GoloAssignment<?>> declarations = new LinkedList<>(); 018 019 /** 020 * Defines a variable declaration (assignment) local to this expression. 021 */ 022 public T with(Object a) { 023 if (!(a instanceof GoloAssignment)) { 024 throw new IllegalArgumentException(("Must be an assignment, got " + a)); 025 } 026 GoloAssignment<?> declaration = (GoloAssignment<?>) a; 027 declarations.add(declaration.declaring()); 028 makeParentOf(declaration); 029 return self(); 030 } 031 032 /** 033 * Returns the local declarations of this expression if any. 034 */ 035 public GoloAssignment<?>[] declarations() { 036 return declarations.toArray(new GoloAssignment<?>[declarations.size()]); 037 } 038 039 /** 040 * Checks if this expression has local variable declarations. 041 */ 042 public boolean hasLocalDeclarations() { 043 return !declarations.isEmpty(); 044 } 045 046 /** 047 * Removes all local declarations. 048 */ 049 public void clearDeclarations() { 050 declarations.clear(); 051 } 052 053 /** 054 * Creates an binary operation representing the anonymous call on this expression. 055 * 056 * <p>For instance, code such as: 057 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 058 * (|x| -> x + 2)(40) 059 * </code></pre> 060 * actually creates a {@link BinaryOperation} or type {@link OperatorType#ANON_CALL} 061 * whose first argument is a {@code ClosureReference} and the second one a {@link FunctionInvocation} holding the 062 * argument. 063 * 064 * <p>This method ease the creation of such a node. 065 * This expression must return a {@link gololang.FunctionReference} when evaluated. This is <em>not checked</em>. 066 * 067 * <p>For instance, in code like 068 * <pre class="listing"><code class="lang-golo" data-lang="golo"> 069 * let f = |x| -> |y| -> x * y 070 * f(2)(42) 071 * </code></pre> 072 * the call is an anonymous call between {@code f(2)} and {@code (42)}, and can be generated by: 073 * <pre class="listing"><code class="lang-java" data-lang="java"> 074 * call("f").withArgs(constant(2)).call(constant(42)) 075 * </pre></code> 076 * <p>See also the 077 * <a href="http://golo-lang.org/documentation/next/#_calling_functions_that_return_functions"> 078 * Golo guide</a>. 079 */ 080 public BinaryOperation call(Object... arguments) { 081 FunctionInvocation invocation; 082 if (arguments.length == 1 && arguments[0] instanceof FunctionInvocation) { 083 invocation = (FunctionInvocation) arguments[0]; 084 if (!invocation.isAnonymous()) { 085 throw new IllegalArgumentException("Invocation in anonymous calls must be anonymous."); 086 } 087 } else { 088 invocation = FunctionInvocation.of(null).withArgs(arguments); 089 } 090 return BinaryOperation.create(OperatorType.ANON_CALL, this, invocation); 091 } 092 093 /** 094 * Expression coercion. 095 * 096 * <p>If the given value is an expression, casts it. If it's a literal value, returns a {@code ConstantStatement}. 097 * If it's a {@link LocalReference}, creates a {@link ReferenceLookup} from it. 098 * 099 * @see ConstantStatement#isLiteralValue(Object) 100 */ 101 public static ExpressionStatement<?> of(Object expr) { 102 if (expr instanceof ExpressionStatement) { 103 return (ExpressionStatement<?>) expr; 104 } 105 if (ConstantStatement.isLiteralValue(expr)) { 106 return ConstantStatement.of(expr); 107 } 108 if (expr instanceof LocalReference) { 109 return ((LocalReference) expr).lookup(); 110 } 111 throw cantConvert("ExpressionStatement", expr); 112 } 113}