/* * Licensed to Elasticsearch under one or more contributor * license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright * ownership. Elasticsearch licenses this file to you under * the Apache License, Version 2.0 (the "License"); you may * not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ package org.elasticsearch.painless.node; import org.elasticsearch.painless.DefBootstrap; import org.elasticsearch.painless.Definition; import org.elasticsearch.painless.Globals; import org.elasticsearch.painless.Locals; import org.elasticsearch.painless.Location; import org.elasticsearch.painless.MethodWriter; import org.objectweb.asm.Type; import java.util.ArrayList; import java.util.List; import java.util.Objects; import java.util.Set; /** * Represents a method call made on a def type. (Internal only.) */ final class PSubDefCall extends AExpression { private final String name; private final List<AExpression> arguments; private StringBuilder recipe = null; private List<String> pointers = new ArrayList<>(); PSubDefCall(Location location, String name, List<AExpression> arguments) { super(location); this.name = Objects.requireNonNull(name); this.arguments = Objects.requireNonNull(arguments); } @Override void extractVariables(Set<String> variables) { throw createError(new IllegalStateException("Illegal tree structure.")); } @Override void analyze(Locals locals) { recipe = new StringBuilder(); int totalCaptures = 0; for (int argument = 0; argument < arguments.size(); ++argument) { AExpression expression = arguments.get(argument); expression.internal = true; expression.analyze(locals); if (expression instanceof ILambda) { ILambda lambda = (ILambda) expression; pointers.add(lambda.getPointer()); // encode this parameter as a deferred reference char ch = (char) (argument + totalCaptures); recipe.append(ch); totalCaptures += lambda.getCaptureCount(); } expression.expected = expression.actual; arguments.set(argument, expression.cast(locals)); } actual = expected == null || explicit ? Definition.DEF_TYPE : expected; } @Override void write(MethodWriter writer, Globals globals) { writer.writeDebugInfo(location); List<Type> parameterTypes = new ArrayList<>(); // first parameter is the receiver, we never know its type: always Object parameterTypes.add(Definition.DEF_TYPE.type); // append each argument for (AExpression argument : arguments) { parameterTypes.add(argument.actual.type); if (argument instanceof ILambda) { ILambda lambda = (ILambda) argument; for (Type capture : lambda.getCaptures()) { parameterTypes.add(capture); } } argument.write(writer, globals); } // create method type from return value and arguments Type methodType = Type.getMethodType(actual.type, parameterTypes.toArray(new Type[0])); List<Object> args = new ArrayList<>(); args.add(recipe.toString()); args.addAll(pointers); writer.invokeDefCall(name, methodType, DefBootstrap.METHOD_CALL, args.toArray()); } @Override public String toString() { return singleLineToStringWithOptionalArgs(arguments, prefix, name); } }