| // Copyright 2014 The Bazel Authors. All rights reserved. |
| // |
| // Licensed 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 com.google.devtools.build.lib.syntax; |
| |
| import com.google.common.collect.ImmutableList; |
| import com.google.common.collect.Iterables; |
| import com.google.devtools.build.lib.events.Location.LineAndColumn; |
| import com.google.devtools.build.lib.profiler.Profiler; |
| import com.google.devtools.build.lib.profiler.ProfilerTask; |
| import com.google.devtools.build.lib.vfs.PathFragment; |
| |
| /** |
| * The actual function registered in the environment. This function is defined in the |
| * parsed code using {@link FunctionDefStatement}. |
| */ |
| public class UserDefinedFunction extends BaseFunction { |
| |
| private final ImmutableList<Statement> statements; |
| |
| // we close over the globals at the time of definition |
| private final Environment.Frame definitionGlobals; |
| |
| protected UserDefinedFunction(Identifier function, |
| FunctionSignature.WithValues<Object, SkylarkType> signature, |
| ImmutableList<Statement> statements, Environment.Frame definitionGlobals) { |
| super(function.getName(), signature, function.getLocation()); |
| this.statements = statements; |
| this.definitionGlobals = definitionGlobals; |
| } |
| |
| public FunctionSignature.WithValues<Object, SkylarkType> getFunctionSignature() { |
| return signature; |
| } |
| |
| ImmutableList<Statement> getStatements() { |
| return statements; |
| } |
| |
| @Override |
| public Object call(Object[] arguments, FuncallExpression ast, Environment env) |
| throws EvalException, InterruptedException { |
| if (!env.mutability().isMutable()) { |
| throw new EvalException(getLocation(), "Trying to call in frozen environment"); |
| } |
| if (env.getStackTrace().contains(this)) { |
| throw new EvalException(getLocation(), |
| String.format("Recursion was detected when calling '%s' from '%s'", |
| getName(), Iterables.getLast(env.getStackTrace()).getName())); |
| } |
| |
| Profiler.instance().startTask(ProfilerTask.SKYLARK_USER_FN, |
| getLocationPathAndLine() + "#" + getName()); |
| try { |
| env.enterScope(this, ast, definitionGlobals); |
| ImmutableList<String> names = signature.getSignature().getNames(); |
| |
| // Registering the functions's arguments as variables in the local Environment |
| int i = 0; |
| for (String name : names) { |
| env.update(name, arguments[i++]); |
| } |
| |
| try { |
| for (Statement stmt : statements) { |
| stmt.exec(env); |
| } |
| } catch (ReturnStatement.ReturnException e) { |
| return e.getValue(); |
| } |
| return Runtime.NONE; |
| } finally { |
| Profiler.instance().completeTask(ProfilerTask.SKYLARK_USER_FN); |
| env.exitScope(); |
| } |
| } |
| |
| /** |
| * Returns the location (filename:line) of the BaseFunction's definition. |
| * |
| * <p>If such a location is not defined, this method returns an empty string. |
| */ |
| private String getLocationPathAndLine() { |
| if (location == null) { |
| return ""; |
| } |
| |
| StringBuilder builder = new StringBuilder(); |
| PathFragment path = location.getPath(); |
| if (path != null) { |
| builder.append(path.getPathString()); |
| } |
| |
| LineAndColumn position = location.getStartLineAndColumn(); |
| if (position != null) { |
| builder.append(":").append(position.getLine()); |
| } |
| return builder.toString(); |
| } |
| } |