blob: 15fb47eca14d9114d85113469b7084a0de0644dd [file] [log] [blame]
// 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.common.base.Preconditions;
import com.google.devtools.build.lib.actions.Action;
import com.google.devtools.build.lib.actions.ActionOwner;
import com.google.devtools.build.lib.clock.Clock;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadCompatible;
import java.time.Duration;
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;
/** 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 Duration getElapsedTimeNoCheck() {
return Duration.ofNanos(getElapsedTimeNanosNoCheck());
}
private long getElapsedTimeNanosNoCheck() {
return finishNanos - startNanos;
}
/**
* Returns the current critical path for the action.
*
* <p>Critical path is defined as : action_execution_time + max(child_critical_path).
*/
public Duration getAggregatedElapsedTime() {
return Duration.ofNanos(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", getElapsedTimeNoCheck().toMillis() / 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));
}
}