|  | // 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.runtime; | 
|  |  | 
|  | import com.google.devtools.build.lib.actions.Action; | 
|  | import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadCompatible; | 
|  | import com.google.devtools.build.lib.util.Clock; | 
|  | import com.google.devtools.build.lib.util.Preconditions; | 
|  | import java.util.concurrent.TimeUnit; | 
|  | import javax.annotation.Nullable; | 
|  |  | 
|  | /** | 
|  | * This class records the critical path for the graph of actions executed. | 
|  | */ | 
|  | @ThreadCompatible | 
|  | public class AbstractCriticalPathComponent<C extends AbstractCriticalPathComponent<C>> { | 
|  |  | 
|  | // These two fields are values of BlazeClock.nanoTime() at the relevant points in time. | 
|  | private long startNanos; | 
|  | private long finishNanos = 0; | 
|  | protected volatile boolean isRunning = true; | 
|  |  | 
|  | /** We keep here the critical path time for the most expensive child. */ | 
|  | private long childAggregatedElapsedTime = 0; | 
|  |  | 
|  | /** The action for which we are storing the stat. */ | 
|  | private final Action action; | 
|  |  | 
|  | /** | 
|  | * Child with the maximum critical path. | 
|  | */ | 
|  | @Nullable | 
|  | private C child; | 
|  |  | 
|  | public AbstractCriticalPathComponent(Action action, long startNanos) { | 
|  | this.action = action; | 
|  | this.startNanos = startNanos; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Record the elapsed time in case the new duration is greater. This method could be called | 
|  | * multiple times if we run shared action concurrently and the one that really gets executed takes | 
|  | * more time to send the finish event and the one that was a cache hit manages to send the event | 
|  | * before. In this case we overwrite the time with the greater time. | 
|  | * | 
|  | * <p>This logic is known to be incorrect, as other actions that depend on this action will not | 
|  | * necessarily use the correct getElapsedTimeNanos(). But we do not want to block action execution | 
|  | * because of this. So in certain conditions we might see another path as the critical path. | 
|  | */ | 
|  | public synchronized boolean finishActionExecution(long startNanos, long finishNanos) { | 
|  | if (isRunning || finishNanos - startNanos > getElapsedTimeNanos()) { | 
|  | this.startNanos = startNanos; | 
|  | this.finishNanos = finishNanos; | 
|  | isRunning = false; | 
|  | return true; | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | /** The action for which we are storing the stat. */ | 
|  | public Action getAction() { | 
|  | return action; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Add statistics for one dependency of this action. Caller should ensure {@code dep} not | 
|  | * running. | 
|  | */ | 
|  | synchronized void addDepInfo(C dep) { | 
|  | long childAggregatedWallTime = dep.getAggregatedElapsedTimeNanos(); | 
|  | // Replace the child if its critical path had the maximum elapsed time. | 
|  | if (child == null || childAggregatedWallTime > this.childAggregatedElapsedTime) { | 
|  | this.childAggregatedElapsedTime = childAggregatedWallTime; | 
|  | child = dep; | 
|  | } | 
|  | } | 
|  |  | 
|  | public long getElapsedTimeMillis() { | 
|  | return TimeUnit.NANOSECONDS.toMillis(getElapsedTimeNanos()); | 
|  | } | 
|  |  | 
|  | long getStartNanos() { | 
|  | return startNanos; | 
|  | } | 
|  |  | 
|  | long getElapsedTimeNanos() { | 
|  | Preconditions.checkState(!isRunning, "Still running %s", action); | 
|  | return finishNanos - startNanos; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the current critical path for the action in nanoseconds. | 
|  | * | 
|  | * <p>Critical path is defined as : action_execution_time + max(child_critical_path). | 
|  | */ | 
|  | public long getAggregatedElapsedTimeMillis() { | 
|  | return TimeUnit.NANOSECONDS.toMillis(getAggregatedElapsedTimeNanos()); | 
|  | } | 
|  |  | 
|  | long getAggregatedElapsedTimeNanos() { | 
|  | Preconditions.checkState(!isRunning, "Still running %s", action); | 
|  | return getElapsedTimeNanos() + childAggregatedElapsedTime; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Get the child critical path component. | 
|  | * | 
|  | * <p>The component dependency with the maximum total critical path time. | 
|  | */ | 
|  | @Nullable | 
|  | public C getChild() { | 
|  | return child; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns a human readable representation of the critical path stats with all the details. | 
|  | */ | 
|  | @Override | 
|  | public String toString() { | 
|  | String currentTime = "still running "; | 
|  | if (!isRunning) { | 
|  | currentTime = String.format("%.2f", getElapsedTimeMillis() / 1000.0) + "s "; | 
|  | } | 
|  | return currentTime + action.describe(); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * When {@code clock} is the same {@link Clock} that was used for computing | 
|  | * {@link #relativeStartNanos}, it returns the wall time since epoch representing when | 
|  | * the action was started. | 
|  | */ | 
|  | public long getStartWallTimeMillis(Clock clock) { | 
|  | long millis = clock.currentTimeMillis(); | 
|  | long nanoElapsed = clock.nanoTime(); | 
|  | return millis - TimeUnit.NANOSECONDS.toMillis((nanoElapsed - startNanos)); | 
|  | } | 
|  | } | 
|  |  |