// 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.actions; import com.google.common.annotations.VisibleForTesting; import com.google.common.base.MoreObjects; import com.google.common.base.MoreObjects.ToStringHelper; import com.google.common.base.Preconditions; import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableMap; import com.google.common.collect.Maps; import com.google.devtools.build.lib.actions.MutableActionGraph.ActionConflictException; import com.google.devtools.build.lib.cmdline.Label; import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe; import com.google.devtools.build.lib.skyframe.serialization.ObjectCodec; import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec; import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec.Strategy; import com.google.devtools.build.skyframe.SkyKey; import com.google.devtools.build.skyframe.SkyValue; import java.util.ArrayList; import java.util.List; import java.util.Map; import javax.annotation.Nullable; /** * Base class for all values which can provide the generating action of an artifact. The primary * instance of such lookup values is {@code ConfiguredTargetValue}. Values that hold the generating * actions of target completion values and build info artifacts also fall into this category. */ public class ActionLookupValue implements SkyValue { protected final List actions; private final ImmutableMap generatingActionIndex; private static Actions.GeneratingActions filterSharedActionsAndThrowRuntimeIfConflict( ActionKeyContext actionKeyContext, List actions) { try { return Actions.filterSharedActionsAndThrowActionConflict(actionKeyContext, actions); } catch (ActionConflictException e) { // Programming bug. throw new IllegalStateException(e); } } @VisibleForTesting public ActionLookupValue( ActionKeyContext actionKeyContext, List actions, boolean removeActionsAfterEvaluation) { this( filterSharedActionsAndThrowRuntimeIfConflict(actionKeyContext, actions), removeActionsAfterEvaluation); } protected ActionLookupValue( ActionKeyContext actionKeyContext, ActionAnalysisMetadata action, boolean removeActionAfterEvaluation) { this(actionKeyContext, ImmutableList.of(action), removeActionAfterEvaluation); } protected ActionLookupValue( Actions.GeneratingActions generatingActions, boolean removeActionsAfterEvaluation) { if (removeActionsAfterEvaluation) { this.actions = new ArrayList<>(generatingActions.getActions()); } else { this.actions = ImmutableList.copyOf(generatingActions.getActions()); } this.generatingActionIndex = generatingActions.getGeneratingActionIndex(); } /** * Returns the action that generates {@code artifact}, if known to this value, or null. This * method should be avoided. Call it only when the action is really needed, and it is known to be * present, either because the execution phase has not started, or because {@link * Action#canRemoveAfterExecution} is known to be false for the action being requested. */ @Nullable public ActionAnalysisMetadata getGeneratingActionDangerousReadJavadoc(Artifact artifact) { Integer actionIndex = getGeneratingActionIndex(artifact); if (actionIndex == null) { return null; } return getActionAnalysisMetadata(actionIndex); } /** * Returns the index of the action that generates {@code artifact} in this value, or null if this * value does not have a generating action for this artifact. The index together with the key for * this {@link ActionLookupValue} uniquely identifies the action. * *

Unlike {@link #getAction}, this may be called after action execution. */ @Nullable public Integer getGeneratingActionIndex(Artifact artifact) { return generatingActionIndex.get(artifact); } /** * Returns the {@link Action} with index {@code index} in this value. Never null. Should only be * called during action execution by {@code ArtifactFunction} and {@code ActionExecutionFunction} * -- after an action has executed, calling this with its index may crash. */ @SuppressWarnings("unchecked") // We test to make sure it's an Action. public Action getAction(int index) { ActionAnalysisMetadata result = getActionAnalysisMetadata(index); Preconditions.checkState(result instanceof Action, "Not action: %s %s %s", result, index, this); return (Action) result; } private ActionAnalysisMetadata getActionAnalysisMetadata(int index) { return Preconditions.checkNotNull(actions.get(index), "null action: %s %s", index, this); } /** * Returns the {@link ActionAnalysisMetadata} at index {@code index} if it is present and * not an {@link Action}. Tree artifacts need their {@code ActionTemplate}s in order to * generate the correct actions, but in general most actions are not needed after they are * executed and may not even be available. */ public ActionAnalysisMetadata getIfPresentAndNotAction(int index) { ActionAnalysisMetadata actionAnalysisMetadata = actions.get(index); if (!(actionAnalysisMetadata instanceof Action)) { return actionAnalysisMetadata; } return null; } /** To be used only when checking consistency of the action graph -- not by other values. */ public Map getMapForConsistencyCheck() { return getMapForConsistencyCheck(generatingActionIndex, actions); } protected ToStringHelper getStringHelper() { return MoreObjects.toStringHelper(this) .add("actions", actions) .add("generatingActionIndex", generatingActionIndex); } @Override public String toString() { return getStringHelper().toString(); } public int getNumActions() { return actions.size(); } public static Map getMapForConsistencyCheck( Map generatingActionIndex, final List actions) { return Maps.transformValues(generatingActionIndex, actions::get); } /** * If this object was initialized with {@code removeActionsAfterEvaluation} and {@link * Action#canRemoveAfterExecution()} is true for {@code action}, then remove this action from this * object's index as a memory-saving measure. The {@code artifact -> index} mapping remains * intact, so this action's execution value can still be addressed by its inputs. */ @ThreadSafe public void actionEvaluated(int actionIndex, Action action) { if (!action.canRemoveAfterExecution()) { return; } if (actions instanceof ArrayList) { // This method may concurrently mutate an ArrayList, which is unsafe on its face. However, // ArrayList mutation on different indices that does not affect the size of the ArrayList is // safe, and that is what does this code does. ArrayList actionArrayList = (ArrayList) actions; ActionAnalysisMetadata oldAction = actionArrayList.set(actionIndex, null); Preconditions.checkState( action.equals(oldAction), "Not same: %s %s %s %s", action, oldAction, this, actionIndex); } } /** * All subclasses of ActionLookupValue "own" artifacts with {@link ArtifactOwner}s that are * subclasses of ActionLookupKey. This allows callers to easily find the value key, while * remaining agnostic to what ActionLookupValues actually exist. */ @AutoCodec(strategy = Strategy.POLYMORPHIC) public abstract static class ActionLookupKey implements ArtifactOwner, SkyKey { public static final ObjectCodec CODEC = new ActionLookupValue_ActionLookupKey_AutoCodec(); @Override public Label getLabel() { return null; } } }