blob: 12acec40f20f2b0b9f0e6364aa5419e1630df21d [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// The synthetic delay framework makes it possible to dynamically inject
// arbitrary delays into into different parts of the codebase. This can be used,
// for instance, for testing various task scheduling algorithms.
//
// The delays are specified in terms of a target duration for a given block of
// code. If the code executes faster than the duration, the thread is made to
// sleep until the deadline is met.
//
// Code can be instrumented for delays with two sets of macros. First, for
// delays that should apply within a scope, use the following macro:
//
// TRACE_EVENT_SYNTHETIC_DELAY("cc.LayerTreeHost.DrawAndSwap");
//
// For delaying operations that span multiple scopes, use:
//
// TRACE_EVENT_SYNTHETIC_DELAY_BEGIN("cc.Scheduler.BeginMainFrame");
// ...
// TRACE_EVENT_SYNTHETIC_DELAY_END("cc.Scheduler.BeginMainFrame");
//
// Here BEGIN establishes the start time for the delay and END executes the
// delay based on the remaining time. If BEGIN is called multiple times in a
// row, END should be called a corresponding number of times. Only the last
// call to END will have an effect.
//
// Note that a single delay may begin on one thread and end on another. This
// implies that a single delay cannot not be applied in several threads at once.
#ifndef KUDU_UTIL_DEBUG_TRACE_EVENT_SYNTHETIC_DELAY_H_
#define KUDU_UTIL_DEBUG_TRACE_EVENT_SYNTHETIC_DELAY_H_
#include <string>
#include "kudu/gutil/atomicops.h"
#include "kudu/gutil/macros.h"
#include "kudu/util/debug/trace_event.h"
#include "kudu/util/monotime.h"
#include "kudu/util/mutex.h"
// Apply a named delay in the current scope.
#define TRACE_EVENT_SYNTHETIC_DELAY(name) \
static AtomicWord INTERNAL_TRACE_EVENT_UID(impl_ptr) = 0; \
trace_event_internal::ScopedSyntheticDelay INTERNAL_TRACE_EVENT_UID(delay)( \
name, &INTERNAL_TRACE_EVENT_UID(impl_ptr));
// Begin a named delay, establishing its timing start point. May be called
// multiple times as long as the calls to TRACE_EVENT_SYNTHETIC_DELAY_END are
// balanced. Only the first call records the timing start point.
#define TRACE_EVENT_SYNTHETIC_DELAY_BEGIN(name) \
do { \
static AtomicWord impl_ptr = 0; \
trace_event_internal::GetOrCreateDelay(name, &impl_ptr)->Begin(); \
} while (false)
// End a named delay. The delay is applied only if this call matches the
// first corresponding call to TRACE_EVENT_SYNTHETIC_DELAY_BEGIN with the
// same delay.
#define TRACE_EVENT_SYNTHETIC_DELAY_END(name) \
do { \
static AtomicWord impl_ptr = 0; \
trace_event_internal::GetOrCreateDelay(name, &impl_ptr)->End(); \
} while (false)
namespace kudu {
namespace debug {
// Time source for computing delay durations. Used for testing.
class TraceEventSyntheticDelayClock {
public:
TraceEventSyntheticDelayClock();
virtual ~TraceEventSyntheticDelayClock();
virtual MonoTime Now() = 0;
private:
DISALLOW_COPY_AND_ASSIGN(TraceEventSyntheticDelayClock);
};
// Single delay point instance.
class TraceEventSyntheticDelay {
public:
enum Mode {
STATIC, // Apply the configured delay every time.
ONE_SHOT, // Apply the configured delay just once.
ALTERNATING // Apply the configured delay every other time.
};
// Returns an existing named delay instance or creates a new one with |name|.
static TraceEventSyntheticDelay* Lookup(const std::string& name);
void SetTargetDuration(const MonoDelta& target_duration);
void SetMode(Mode mode);
void SetClock(TraceEventSyntheticDelayClock* clock);
// Begin the delay, establishing its timing start point. May be called
// multiple times as long as the calls to End() are balanced. Only the first
// call records the timing start point.
void Begin();
// End the delay. The delay is applied only if this call matches the first
// corresponding call to Begin() with the same delay.
void End();
// Begin a parallel instance of the delay. Several parallel instances may be
// active simultaneously and will complete independently. The computed end
// time for the delay is stored in |out_end_time|, which should later be
// passed to EndParallel().
void BeginParallel(MonoTime* out_end_time);
// End a previously started parallel delay. |end_time| is the delay end point
// computed by BeginParallel().
void EndParallel(const MonoTime& end_time);
private:
TraceEventSyntheticDelay();
~TraceEventSyntheticDelay();
friend class TraceEventSyntheticDelayRegistry;
void Initialize(const std::string& name,
TraceEventSyntheticDelayClock* clock);
MonoTime CalculateEndTimeLocked(const MonoTime& start_time);
void ApplyDelay(const MonoTime& end_time);
Mutex lock_;
Mode mode_;
std::string name_;
int begin_count_;
int trigger_count_;
MonoTime end_time_;
MonoDelta target_duration_;
TraceEventSyntheticDelayClock* clock_;
DISALLOW_COPY_AND_ASSIGN(TraceEventSyntheticDelay);
};
// Set the target durations of all registered synthetic delay points to zero.
void ResetTraceEventSyntheticDelays();
} // namespace debug
} // namespace kudu
namespace trace_event_internal {
// Helper class for scoped delays. Do not use directly.
class ScopedSyntheticDelay {
public:
explicit ScopedSyntheticDelay(const char* name,
AtomicWord* impl_ptr);
~ScopedSyntheticDelay();
private:
kudu::debug::TraceEventSyntheticDelay* delay_impl_;
kudu::MonoTime end_time_;
DISALLOW_COPY_AND_ASSIGN(ScopedSyntheticDelay);
};
// Helper for registering delays. Do not use directly.
kudu::debug::TraceEventSyntheticDelay*
GetOrCreateDelay(const char* name, AtomicWord* impl_ptr);
} // namespace trace_event_internal
#endif /* KUDU_UTIL_DEBUG_TRACE_EVENT_SYNTHETIC_DELAY_H_ */