com.google.devtools.build.lib.syntax.UserDefinedFunction.java Source code

Java tutorial

Introduction

Here is the source code for com.google.devtools.build.lib.syntax.UserDefinedFunction.java

Source

// 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.profiler.Profiler;
import com.google.devtools.build.lib.profiler.ProfilerTask;

/**
 * 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) throws EvalException {
        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, 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) {
                    if (stmt instanceof ReturnStatement) {
                        // Performance optimization.
                        // Executing the statement would throw an exception, which is slow.
                        return ((ReturnStatement) stmt).getReturnExpression().eval(env);
                    } else {
                        stmt.exec(env);
                    }
                }
            } catch (ReturnStatement.ReturnException e) {
                return e.getValue();
            }
            return Runtime.NONE;
        } finally {
            Profiler.instance().completeTask(ProfilerTask.SKYLARK_USER_FN);
            env.exitScope();
        }
    }
}