| // 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.Function; |
| import com.google.common.base.MoreObjects; |
| import com.google.common.base.MoreObjects.ToStringHelper; |
| 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.util.Preconditions; |
| import com.google.devtools.build.skyframe.LegacySkyKey; |
| import com.google.devtools.build.skyframe.SkyFunctionName; |
| 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<ActionAnalysisMetadata> actions; |
| private final ImmutableMap<Artifact, Integer> generatingActionIndex; |
| |
| private static Actions.GeneratingActions filterSharedActionsAndThrowRuntimeIfConflict( |
| List<ActionAnalysisMetadata> actions) { |
| try { |
| return Actions.filterSharedActionsAndThrowActionConflict(actions); |
| } catch (ActionConflictException e) { |
| // Programming bug. |
| throw new IllegalStateException(e); |
| } |
| } |
| |
| @VisibleForTesting |
| public ActionLookupValue( |
| List<ActionAnalysisMetadata> actions, boolean removeActionsAfterEvaluation) { |
| this(filterSharedActionsAndThrowRuntimeIfConflict(actions), removeActionsAfterEvaluation); |
| } |
| |
| protected ActionLookupValue(ActionAnalysisMetadata action, boolean removeActionAfterEvaluation) { |
| this(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. |
| * |
| * <p>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 |
| * <i>not</i> 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<Artifact, ActionAnalysisMetadata> 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(); |
| } |
| |
| @VisibleForTesting |
| public static SkyKey key(ActionLookupKey ownerKey) { |
| return ownerKey.getSkyKey(); |
| } |
| |
| public int getNumActions() { |
| return actions.size(); |
| } |
| |
| public static Map<Artifact, ActionAnalysisMetadata> getMapForConsistencyCheck( |
| Map<Artifact, Integer> generatingActionIndex, |
| final List<? extends ActionAnalysisMetadata> actions) { |
| return Maps.transformValues( |
| generatingActionIndex, |
| new Function<Integer, ActionAnalysisMetadata>() { |
| @Override |
| public ActionAnalysisMetadata apply(Integer index) { |
| return actions.get(index); |
| } |
| }); |
| } |
| |
| /** |
| * 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<ActionAnalysisMetadata> actionArrayList = |
| (ArrayList<ActionAnalysisMetadata>) actions; |
| ActionAnalysisMetadata oldAction = actionArrayList.set(actionIndex, null); |
| Preconditions.checkState( |
| action.equals(oldAction), "Not same: %s %s %s %s", action, oldAction, this, actionIndex); |
| } |
| } |
| |
| /** |
| * ArtifactOwner is not a SkyKey, but we wish to convert any ArtifactOwner into a SkyKey as simply |
| * as possible. To that end, all subclasses of ActionLookupValue "own" artifacts with |
| * ArtifactOwners that are subclasses of ActionLookupKey. This allows callers to easily find the |
| * value key, while remaining agnostic to what ActionLookupValues actually exist. |
| * |
| * <p>The methods of this class should only be called by {@link ActionLookupValue#key}. |
| */ |
| public abstract static class ActionLookupKey implements ArtifactOwner { |
| @Override |
| public Label getLabel() { |
| return null; |
| } |
| |
| /** |
| * Subclasses must override this to specify their specific value type, unless they override |
| * {@link #getSkyKey}, in which case they are free not to implement this method. |
| */ |
| protected abstract SkyFunctionName getType(); |
| |
| protected SkyKey getSkyKeyInternal() { |
| return LegacySkyKey.create(getType(), this); |
| } |
| |
| /** |
| * Prefer {@link ActionLookupValue#key} to calling this method directly. |
| * |
| * <p>Subclasses may override {@link #getSkyKeyInternal} if the {@link SkyKey} argument should |
| * not be this {@link ActionLookupKey} itself. |
| */ |
| public final SkyKey getSkyKey() { |
| SkyKey result = getSkyKeyInternal(); |
| Preconditions.checkState( |
| result.argument() instanceof ActionLookupKey, |
| "Not ActionLookupKey for %s: %s", |
| this, |
| result); |
| return result; |
| } |
| } |
| } |