// 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.actions.ActionOwner; import com.google.devtools.build.lib.cmdline.Label; 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; volatile boolean isRunning = true; /** We keep here the critical path time for the most expensive child. */ private long childAggregatedElapsedTime = 0; /** May be nulled out after finished running to allow the action to be GC'ed. */ @Nullable protected 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. May be null if the action has finished running. */ @Nullable public final Action maybeGetAction() { return action; } public String prettyPrintAction() { return getActionNotNull().prettyPrint(); } @Nullable public Label getOwner() { ActionOwner owner = getActionNotNull().getOwner(); if (owner != null && owner.getLabel() != null) { return owner.getLabel(); } return null; } public String getMnemonic() { return getActionNotNull().getMnemonic(); } private Action getActionNotNull() { return Preconditions.checkNotNull(action, this); } /** * 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", this); return getElapsedTimeNanosNoCheck(); } /** To be used only in debugging: skips state invariance checks to avoid crash-looping. */ protected long getElapsedTimeMillisNoCheck() { return TimeUnit.NANOSECONDS.toMillis(getElapsedTimeNanosNoCheck()); } private long getElapsedTimeNanosNoCheck() { 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", this); 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 string representation of the action. Only for use in crash messages and the like. */ protected String getActionString() { Action action = maybeGetAction(); return (action == null ? "(null action)" : action.prettyPrint()); } /** * 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", getElapsedTimeMillisNoCheck() / 1000.0) + "s "; } return currentTime + getActionString(); } /** * 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)); } }