| // Copyright 2015 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.profiler; |
| |
| import com.google.common.base.Preconditions; |
| import com.google.devtools.build.lib.clock.BlazeClock; |
| import com.google.devtools.build.lib.clock.Clock; |
| import java.time.Duration; |
| import java.util.concurrent.TimeUnit; |
| import java.util.concurrent.atomic.AtomicBoolean; |
| import java.util.concurrent.atomic.AtomicReference; |
| import java.util.logging.Logger; |
| import javax.annotation.Nullable; |
| |
| /** |
| * A convenient way to actively get access to timing information (e.g. for logging and/or |
| * profiling purposes) with minimal boilerplate. The lack of boilerplate comes at a performance |
| * cost; do not use {@link AutoProfiler} on performance critical code. |
| * |
| * <p>The intended usage is: |
| * |
| * <pre> |
| * {@code |
| * try (AutoProfiler p = AutoProfiler.logged("<description of your code>")) { |
| * // Your code here. |
| * } |
| * } |
| * </pre> |
| * |
| * <p>but if the try-with-resources pattern is too cumbersome, you can also do |
| * |
| * <pre> |
| * {@code |
| * AutoProfiler p = AutoProfiler.logged("<description of your code>"); |
| * // Your code here. |
| * long elapsedTimeNanos = p.completeAndGetElapsedTimeNanos(); |
| * } |
| * </pre> |
| * |
| * <p>An {@link AutoProfiler} can also automatically talk to the active {@link Profiler} instance: |
| * |
| * <pre> |
| * {@code |
| * try (AutoProfiler p = AutoProfiler.profiled("<description of your code>")) { |
| * // Your code here. |
| * } |
| * } |
| * </pre> |
| */ |
| public class AutoProfiler implements SilentCloseable { |
| private static final AtomicReference<Clock> CLOCK_REF = new AtomicReference<>(null); |
| private static final AtomicReference<LoggingElapsedTimeReceiverFactory> |
| LOGGING_ELAPSED_TIME_RECEIVER_FACTORY_REF = new AtomicReference<>( |
| LoggingElapsedTimeReceiver.FACTORY); |
| |
| private final ElapsedTimeReceiver elapsedTimeReceiver; |
| private final long startTimeNanos; |
| private final AtomicBoolean closed = new AtomicBoolean(false); |
| |
| private AutoProfiler(ElapsedTimeReceiver elapsedTimeReceiver, long startTimeNanos) { |
| this.elapsedTimeReceiver = elapsedTimeReceiver; |
| this.startTimeNanos = startTimeNanos; |
| } |
| |
| /** A opaque receiver of elapsed time information. */ |
| public interface ElapsedTimeReceiver { |
| /** |
| * Receives the elapsed time of the lifetime of an {@link AutoProfiler} instance. |
| * |
| * <p>Note that System#nanoTime isn't guaranteed to be non-decreasing, so implementations should |
| * check for non-positive {@code elapsedTimeNanos} if they care about this sort of thing. |
| */ |
| void accept(long elapsedTimeNanos); |
| } |
| |
| /** |
| * Sets the clock to use. May be called before trying to call any of the other static methods; |
| * otherwise {@link BlazeClock#instance} will be used. |
| * |
| * <p>Calling this more than once (e.g. after calling any of the other static methods) results in |
| * unspecified behavior. |
| */ |
| public static void setClock(Clock clock) { |
| CLOCK_REF.set(clock); |
| } |
| |
| private static long nanoTime() { |
| Clock clock = CLOCK_REF.get(); |
| return clock != null ? clock.nanoTime() : BlazeClock.instance().nanoTime(); |
| } |
| |
| /** |
| * Factory to use for creating {@link ElapsedTimeReceiver} instances for servicing |
| * {@link #logged} et al. |
| */ |
| public interface LoggingElapsedTimeReceiverFactory { |
| ElapsedTimeReceiver create( |
| String taskDescription, Logger logger, long minTimeForLogging, TimeUnit timeUnit); |
| } |
| |
| /** |
| * Inject a custom {@link LoggingElapsedTimeReceiverFactory} for servicing {@link #logged} et al. |
| * |
| * <p>Use this if you want custom logging behavior. |
| */ |
| public static void setLoggingElapsedTimeReceiverFactory( |
| LoggingElapsedTimeReceiverFactory factory) { |
| LOGGING_ELAPSED_TIME_RECEIVER_FACTORY_REF.set(factory); |
| } |
| |
| /** |
| * Returns an {@link AutoProfiler} that, when closed, logs the elapsed time in milliseconds to |
| * the given {@link Logger}. |
| * |
| * <p>The returned {@link AutoProfiler} is thread-safe. |
| */ |
| public static AutoProfiler logged(String taskDescription, Logger logger) { |
| return logged(taskDescription, logger, TimeUnit.MILLISECONDS); |
| } |
| |
| /** |
| * Returns an {@link AutoProfiler} that, when closed, logs the elapsed time in the given |
| * {@link TimeUnit} to the given {@link Logger}. |
| * |
| * <p>The returned {@link AutoProfiler} is thread-safe. |
| */ |
| public static AutoProfiler logged(String taskDescription, Logger logger, TimeUnit timeUnit) { |
| return create( |
| LOGGING_ELAPSED_TIME_RECEIVER_FACTORY_REF |
| .get() |
| .create(taskDescription, logger, 0L, timeUnit)); |
| } |
| |
| /** |
| * Returns an {@link AutoProfiler} that, when closed, logs the elapsed time in milliseconds to the |
| * given {@link Logger} if it is greater than {@code minTimeForLoggingInMilliseconds}. |
| * |
| * <p>The returned {@link AutoProfiler} is thread-safe. |
| */ |
| public static AutoProfiler logged( |
| String taskDescription, Logger logger, long minTimeForLoggingInMilliseconds) { |
| return create( |
| LOGGING_ELAPSED_TIME_RECEIVER_FACTORY_REF |
| .get() |
| .create( |
| taskDescription, logger, minTimeForLoggingInMilliseconds, TimeUnit.MILLISECONDS)); |
| } |
| |
| /** |
| * Returns an {@link AutoProfiler} that, when closed, records the elapsed time using {@link |
| * Profiler}. |
| * |
| * <p>The returned {@link AutoProfiler} is thread-safe. |
| */ |
| public static AutoProfiler profiled(String description, ProfilerTask profilerTaskType) { |
| return create(new ProfilingElapsedTimeReceiver(description, profilerTaskType)); |
| } |
| |
| /** |
| * Returns an {@link AutoProfiler} that, when closed, records the elapsed time using |
| * {@link Profiler} and also logs it (in milliseconds) to the given {@link Logger}. |
| * |
| * <p>The returned {@link AutoProfiler} is thread-safe. |
| */ |
| public static AutoProfiler profiledAndLogged(String taskDescription, |
| ProfilerTask profilerTaskType, Logger logger) { |
| ElapsedTimeReceiver profilingReceiver = |
| new ProfilingElapsedTimeReceiver(taskDescription, profilerTaskType); |
| ElapsedTimeReceiver loggingReceiver = |
| LOGGING_ELAPSED_TIME_RECEIVER_FACTORY_REF |
| .get() |
| .create(taskDescription, logger, 0L, TimeUnit.MILLISECONDS); |
| return create(new SequencedElapsedTimeReceiver(profilingReceiver, loggingReceiver)); |
| } |
| |
| /** |
| * Returns an {@link AutoProfiler} that doesn't do anything when closed and so is only useful for |
| * {@link #completeAndGetElapsedTimeNanos()}. |
| */ |
| public static AutoProfiler timed() { |
| return create(NULL_RECEIVER); |
| } |
| |
| /** |
| * Returns an {@link AutoProfiler} that, when closed, invokes the given |
| * {@link ElapsedTimeReceiver}. |
| * |
| * <p>The returned {@link AutoProfiler} is as thread-safe as the given |
| * {@link ElapsedTimeReceiver} is. |
| */ |
| public static AutoProfiler create(ElapsedTimeReceiver elapsedTimeReceiver) { |
| return new AutoProfiler(elapsedTimeReceiver, nanoTime()); |
| } |
| |
| /** |
| * Manually completes the profiling (useful to trigger the underlying action on completion). |
| * |
| * <p>At most one of {@link #complete}, {@link #completeAndGetElapsedTimeNanos} and {@link #close} |
| * may be called. |
| */ |
| public void complete() { |
| close(); |
| } |
| |
| /** |
| * Manually completes the profiling and returns the elapsed time in nanoseconds. |
| * |
| * <p>At most one of {@link #complete}, {@link #completeAndGetElapsedTimeNanos} and {@link #close} |
| * may be called. |
| */ |
| public long completeAndGetElapsedTimeNanos() { |
| long elapsedTimeNanos = nanoTime() - startTimeNanos; |
| Preconditions.checkState(closed.compareAndSet(false, true)); |
| elapsedTimeReceiver.accept(elapsedTimeNanos); |
| return elapsedTimeNanos; |
| } |
| |
| /** |
| * Automatically completes the profiling. |
| * |
| * <p>At most one of {@link #complete}, {@link #completeAndGetElapsedTimeNanos} and {@link #close} |
| * may be called. |
| */ |
| @Override |
| public void close() { |
| long elapsedTimeNanos = nanoTime() - startTimeNanos; |
| Preconditions.checkState(closed.compareAndSet(false, true)); |
| elapsedTimeReceiver.accept(elapsedTimeNanos); |
| } |
| |
| private static final ElapsedTimeReceiver NULL_RECEIVER = new ElapsedTimeReceiver() { |
| @Override |
| public void accept(long elapsedTimeNanos) { |
| } |
| }; |
| |
| private static class SequencedElapsedTimeReceiver implements ElapsedTimeReceiver { |
| private final ElapsedTimeReceiver firstReceiver; |
| private final ElapsedTimeReceiver secondReceiver; |
| |
| private SequencedElapsedTimeReceiver(ElapsedTimeReceiver firstReceiver, |
| ElapsedTimeReceiver secondReceiver) { |
| this.firstReceiver = firstReceiver; |
| this.secondReceiver = secondReceiver; |
| } |
| |
| @Override |
| public void accept(long elapsedTimeNanos) { |
| firstReceiver.accept(elapsedTimeNanos); |
| secondReceiver.accept(elapsedTimeNanos); |
| } |
| } |
| |
| /** {@link ElapsedTimeReceiver} that will not log a message if the time elapsed is too small. */ |
| public abstract static class FilteringElapsedTimeReceiver implements ElapsedTimeReceiver { |
| private final String taskDescription; |
| private final TimeUnit timeUnit; |
| private final long minNanosForLogging; |
| |
| protected FilteringElapsedTimeReceiver( |
| String taskDescription, long minTimeForLogging, TimeUnit timeUnit) { |
| this.taskDescription = taskDescription; |
| this.timeUnit = timeUnit; |
| this.minNanosForLogging = timeUnit.toNanos(minTimeForLogging); |
| } |
| |
| protected abstract void log(String logMessage); |
| /** |
| * Returns a message about the amount of time spent doing a task if {@code elapsedTimeNanos} is |
| * at least {@link #minNanosForLogging} and null otherwise. |
| */ |
| @Nullable |
| private String loggingMessage(long elapsedTimeNanos) { |
| if (elapsedTimeNanos >= minNanosForLogging) { |
| return String.format( |
| "Spent %d %s doing %s", |
| timeUnit.convert(elapsedTimeNanos, TimeUnit.NANOSECONDS), |
| timeUnit.toString().toLowerCase(), |
| taskDescription); |
| } else { |
| return null; |
| } |
| } |
| |
| @Override |
| public final void accept(long elapsedTimeNanos) { |
| String logMessage = loggingMessage(elapsedTimeNanos); |
| if (logMessage != null) { |
| log(logMessage); |
| } |
| } |
| } |
| |
| /** |
| * {@link ElapsedTimeReceiver} that logs a message in {@link #accept} if the elapsed time is at |
| * least a given threshold. |
| */ |
| public static class LoggingElapsedTimeReceiver extends FilteringElapsedTimeReceiver { |
| private static final LoggingElapsedTimeReceiverFactory FACTORY = |
| new LoggingElapsedTimeReceiverFactory() { |
| @Override |
| public ElapsedTimeReceiver create( |
| String taskDescription, Logger logger, long minTimeForLogging, TimeUnit timeUnit) { |
| return new LoggingElapsedTimeReceiver( |
| taskDescription, logger, minTimeForLogging, timeUnit); |
| } |
| }; |
| |
| protected final Logger logger; |
| |
| protected LoggingElapsedTimeReceiver( |
| String taskDescription, Logger logger, long minTimeForLogging, TimeUnit timeUnit) { |
| super(taskDescription, minTimeForLogging, timeUnit); |
| this.logger = logger; |
| } |
| |
| @Override |
| protected void log(String logMessage) { |
| logger.info(logMessage); |
| } |
| } |
| |
| private static class ProfilingElapsedTimeReceiver implements ElapsedTimeReceiver { |
| private final long startTimeNanos; |
| private final String description; |
| private final ProfilerTask profilerTaskType; |
| |
| private ProfilingElapsedTimeReceiver(String description, ProfilerTask profilerTaskType) { |
| this.startTimeNanos = Profiler.nanoTimeMaybe(); |
| this.description = description; |
| this.profilerTaskType = profilerTaskType; |
| } |
| |
| @Override |
| public void accept(long elapsedTimeNanos) { |
| if (elapsedTimeNanos > 0) { |
| Profiler.instance() |
| .logSimpleTaskDuration( |
| startTimeNanos, Duration.ofNanos(elapsedTimeNanos), profilerTaskType, description); |
| } |
| } |
| } |
| } |
| |