// Copyright 2014 Google Inc. 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 statements; // we close over the globals at the time of definition private final Environment.Frame definitionGlobals; protected UserDefinedFunction(Identifier function, FunctionSignature.WithValues signature, ImmutableList statements, Environment.Frame definitionGlobals) { super(function.getName(), signature, function.getLocation()); this.statements = statements; this.definitionGlobals = definitionGlobals; } public FunctionSignature.WithValues getFunctionSignature() { return signature; } ImmutableList 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()); Statement lastStatement = null; try { env.enterScope(this, ast, definitionGlobals); ImmutableList 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) { lastStatement = stmt; stmt.exec(env); } } catch (ReturnStatement.ReturnException e) { return e.getValue(); } return Runtime.NONE; } catch (EvalExceptionWithStackTrace ex) { // We need this block since the next "catch" must only catch EvalExceptions that don't have a // stack trace yet. throw ex; } catch (EvalException ex) { EvalExceptionWithStackTrace real = new EvalExceptionWithStackTrace(ex, lastStatement.getLocation()); real.registerStatement(lastStatement); throw real; } finally { Profiler.instance().completeTask(ProfilerTask.SKYLARK_USER_FN); env.exitScope(); } } /** * Returns the location (filename:line) of the BaseFunction's definition. * *

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