|  | // Copyright 2015 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. | 
|  |  | 
|  | #ifndef BASE_PROFILER_STACK_SAMPLING_PROFILER_H_ | 
|  | #define BASE_PROFILER_STACK_SAMPLING_PROFILER_H_ | 
|  |  | 
|  | #include <stddef.h> | 
|  |  | 
|  | #include <memory> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "base/atomicops.h" | 
|  | #include "base/base_export.h" | 
|  | #include "base/callback.h" | 
|  | #include "base/files/file_path.h" | 
|  | #include "base/macros.h" | 
|  | #include "base/strings/string16.h" | 
|  | #include "base/synchronization/waitable_event.h" | 
|  | #include "base/threading/platform_thread.h" | 
|  | #include "base/time/time.h" | 
|  |  | 
|  | namespace base { | 
|  |  | 
|  | class NativeStackSampler; | 
|  | class NativeStackSamplerTestDelegate; | 
|  |  | 
|  | // StackSamplingProfiler periodically stops a thread to sample its stack, for | 
|  | // the purpose of collecting information about which code paths are | 
|  | // executing. This information is used in aggregate by UMA to identify hot | 
|  | // and/or janky code paths. | 
|  | // | 
|  | // Sample StackSamplingProfiler usage: | 
|  | // | 
|  | //   // Create and customize params as desired. | 
|  | //   base::StackStackSamplingProfiler::SamplingParams params; | 
|  | //   // Any thread's ID may be passed as the target. | 
|  | //   base::StackSamplingProfiler profiler(base::PlatformThread::CurrentId()), | 
|  | //       params); | 
|  | // | 
|  | //   // Or, to process the profiles within Chrome rather than via UMA, use a | 
|  | //   // custom completed callback: | 
|  | //   base::StackStackSamplingProfiler::CompletedCallback | 
|  | //       thread_safe_callback = ...; | 
|  | //   base::StackSamplingProfiler profiler(base::PlatformThread::CurrentId()), | 
|  | //       params, thread_safe_callback); | 
|  | // | 
|  | //   profiler.Start(); | 
|  | //   // ... work being done on the target thread here ... | 
|  | //   profiler.Stop();  // optional, stops collection before complete per params | 
|  | // | 
|  | // The default SamplingParams causes stacks to be recorded in a single burst at | 
|  | // a 10Hz interval for a total of 30 seconds. All of these parameters may be | 
|  | // altered as desired. | 
|  | // | 
|  | // When all call stack profiles are complete, or the profiler is stopped, the | 
|  | // completed callback is called from a thread created by the profiler with the | 
|  | // collected profiles. | 
|  | // | 
|  | // The results of the profiling are passed to the completed callback and consist | 
|  | // of a vector of CallStackProfiles. Each CallStackProfile corresponds to a | 
|  | // burst as specified in SamplingParams and contains a set of Samples and | 
|  | // Modules. One Sample corresponds to a single recorded stack, and the Modules | 
|  | // record those modules associated with the recorded stack frames. | 
|  | class BASE_EXPORT StackSamplingProfiler { | 
|  | public: | 
|  | // Module represents the module (DLL or exe) corresponding to a stack frame. | 
|  | struct BASE_EXPORT Module { | 
|  | Module(); | 
|  | Module(uintptr_t base_address, | 
|  | const std::string& id, | 
|  | const FilePath& filename); | 
|  | ~Module(); | 
|  |  | 
|  | // Points to the base address of the module. | 
|  | uintptr_t base_address; | 
|  |  | 
|  | // An opaque binary string that uniquely identifies a particular program | 
|  | // version with high probability. This is parsed from headers of the loaded | 
|  | // module. | 
|  | // For binaries generated by GNU tools: | 
|  | //   Contents of the .note.gnu.build-id field. | 
|  | // On Windows: | 
|  | //   GUID + AGE in the debug image headers of a module. | 
|  | std::string id; | 
|  |  | 
|  | // The filename of the module. | 
|  | FilePath filename; | 
|  | }; | 
|  |  | 
|  | // Frame represents an individual sampled stack frame with module information. | 
|  | struct BASE_EXPORT Frame { | 
|  | // Identifies an unknown module. | 
|  | static const size_t kUnknownModuleIndex = static_cast<size_t>(-1); | 
|  |  | 
|  | Frame(uintptr_t instruction_pointer, size_t module_index); | 
|  | ~Frame(); | 
|  |  | 
|  | // Default constructor to satisfy IPC macros. Do not use explicitly. | 
|  | Frame(); | 
|  |  | 
|  | // The sampled instruction pointer within the function. | 
|  | uintptr_t instruction_pointer; | 
|  |  | 
|  | // Index of the module in CallStackProfile::modules. We don't represent | 
|  | // module state directly here to save space. | 
|  | size_t module_index; | 
|  | }; | 
|  |  | 
|  | // Sample represents a set of stack frames with some extra information. | 
|  | struct BASE_EXPORT Sample { | 
|  | Sample(); | 
|  | Sample(const Sample& sample); | 
|  | ~Sample(); | 
|  |  | 
|  | // These constructors are used only during testing. | 
|  | Sample(const Frame& frame); | 
|  | Sample(const std::vector<Frame>& frames); | 
|  |  | 
|  | // The entire stack frame when the sample is taken. | 
|  | std::vector<Frame> frames; | 
|  |  | 
|  | // A bit-field indicating which process milestones have passed. This can be | 
|  | // used to tell where in the process lifetime the samples are taken. Just | 
|  | // as a "lifetime" can only move forward, these bits mark the milestones of | 
|  | // the processes life as they occur. Bits can be set but never reset. The | 
|  | // actual definition of the individual bits is left to the user of this | 
|  | // module. | 
|  | uint32_t process_milestones = 0; | 
|  | }; | 
|  |  | 
|  | // CallStackProfile represents a set of samples. | 
|  | struct BASE_EXPORT CallStackProfile { | 
|  | CallStackProfile(); | 
|  | CallStackProfile(CallStackProfile&& other); | 
|  | ~CallStackProfile(); | 
|  |  | 
|  | CallStackProfile& operator=(CallStackProfile&& other); | 
|  |  | 
|  | CallStackProfile CopyForTesting() const; | 
|  |  | 
|  | std::vector<Module> modules; | 
|  | std::vector<Sample> samples; | 
|  |  | 
|  | // Duration of this profile. | 
|  | TimeDelta profile_duration; | 
|  |  | 
|  | // Time between samples. | 
|  | TimeDelta sampling_period; | 
|  |  | 
|  | private: | 
|  | // Copying is possible but expensive so disallow it except for internal use | 
|  | // (i.e. CopyForTesting); use std::move instead. | 
|  | CallStackProfile(const CallStackProfile& other); | 
|  |  | 
|  | DISALLOW_ASSIGN(CallStackProfile); | 
|  | }; | 
|  |  | 
|  | using CallStackProfiles = std::vector<CallStackProfile>; | 
|  |  | 
|  | // Represents parameters that configure the sampling. | 
|  | struct BASE_EXPORT SamplingParams { | 
|  | // Time to delay before first samples are taken. | 
|  | TimeDelta initial_delay = TimeDelta::FromMilliseconds(0); | 
|  |  | 
|  | // Number of sampling bursts to perform. | 
|  | int bursts = 1; | 
|  |  | 
|  | // Interval between sampling bursts. This is the desired duration from the | 
|  | // start of one burst to the start of the next burst. | 
|  | TimeDelta burst_interval = TimeDelta::FromSeconds(10); | 
|  |  | 
|  | // Number of samples to record per burst. | 
|  | int samples_per_burst = 300; | 
|  |  | 
|  | // Interval between samples during a sampling burst. This is the desired | 
|  | // duration from the start of one sample to the start of the next sample. | 
|  | TimeDelta sampling_interval = TimeDelta::FromMilliseconds(100); | 
|  | }; | 
|  |  | 
|  | // Testing support. These methods are static beause they interact with the | 
|  | // sampling thread, a singleton used by all StackSamplingProfiler objects. | 
|  | // These methods can only be called by the same thread that started the | 
|  | // sampling. | 
|  | class BASE_EXPORT TestAPI { | 
|  | public: | 
|  | // Resets the internal state to that of a fresh start. This is necessary | 
|  | // so that tests don't inherit state from previous tests. | 
|  | static void Reset(); | 
|  |  | 
|  | // Resets internal annotations (like process phase) to initial values. | 
|  | static void ResetAnnotations(); | 
|  |  | 
|  | // Returns whether the sampling thread is currently running or not. | 
|  | static bool IsSamplingThreadRunning(); | 
|  |  | 
|  | // Disables inherent idle-shutdown behavior. | 
|  | static void DisableIdleShutdown(); | 
|  |  | 
|  | // Initiates an idle shutdown task, as though the idle timer had expired, | 
|  | // causing the thread to exit. There is no "idle" check so this must be | 
|  | // called only when all sampling tasks have completed. This blocks until | 
|  | // the task has been executed, though the actual stopping of the thread | 
|  | // still happens asynchronously. Watch IsSamplingThreadRunning() to know | 
|  | // when the thread has exited. If |simulate_intervening_start| is true then | 
|  | // this method will make it appear to the shutdown task that a new profiler | 
|  | // was started between when the idle-shutdown was initiated and when it | 
|  | // runs. | 
|  | static void PerformSamplingThreadIdleShutdown( | 
|  | bool simulate_intervening_start); | 
|  | }; | 
|  |  | 
|  | // The callback type used to collect completed profiles. The passed |profiles| | 
|  | // are move-only. Other threads, including the UI thread, may block on | 
|  | // callback completion so this should run as quickly as possible. | 
|  | // | 
|  | // IMPORTANT NOTE: The callback is invoked on a thread the profiler | 
|  | // constructs, rather than on the thread used to construct the profiler and | 
|  | // set the callback, and thus the callback must be callable on any thread. For | 
|  | // threads with message loops that create StackSamplingProfilers, posting a | 
|  | // task to the message loop with the moved (i.e. std::move) profiles is the | 
|  | // thread-safe callback implementation. | 
|  | using CompletedCallback = Callback<void(CallStackProfiles)>; | 
|  |  | 
|  | // Creates a profiler for the CURRENT thread that sends completed profiles | 
|  | // to |callback|. An optional |test_delegate| can be supplied by tests. | 
|  | // The caller must ensure that this object gets destroyed before the current | 
|  | // thread exits. | 
|  | StackSamplingProfiler( | 
|  | const SamplingParams& params, | 
|  | const CompletedCallback& callback, | 
|  | NativeStackSamplerTestDelegate* test_delegate = nullptr); | 
|  |  | 
|  | // Creates a profiler for ANOTHER thread that sends completed profiles to | 
|  | // |callback|. An optional |test_delegate| can be supplied by tests. | 
|  | // | 
|  | // IMPORTANT: The caller must ensure that the thread being sampled does not | 
|  | // exit before this object gets destructed or Bad Things(tm) may occur. | 
|  | StackSamplingProfiler( | 
|  | PlatformThreadId thread_id, | 
|  | const SamplingParams& params, | 
|  | const CompletedCallback& callback, | 
|  | NativeStackSamplerTestDelegate* test_delegate = nullptr); | 
|  |  | 
|  | // Stops any profiling currently taking place before destroying the profiler. | 
|  | // This will block until the callback has been run if profiling has started | 
|  | // but not already finished. | 
|  | ~StackSamplingProfiler(); | 
|  |  | 
|  | // Initializes the profiler and starts sampling. Might block on a | 
|  | // WaitableEvent if this StackSamplingProfiler was previously started and | 
|  | // recently stopped, while the previous profiling phase winds down. | 
|  | void Start(); | 
|  |  | 
|  | // Stops the profiler and any ongoing sampling. This method will return | 
|  | // immediately with the callback being run asynchronously. At most one | 
|  | // more stack sample will be taken after this method returns. Calling this | 
|  | // function is optional; if not invoked profiling terminates when all the | 
|  | // profiling bursts specified in the SamplingParams are completed or the | 
|  | // profiler object is destroyed, whichever occurs first. | 
|  | void Stop(); | 
|  |  | 
|  | // Set the current system state that is recorded with each captured stack | 
|  | // frame. This is thread-safe so can be called from anywhere. The parameter | 
|  | // value should be from an enumeration of the appropriate type with values | 
|  | // ranging from 0 to 31, inclusive. This sets bits within Sample field of | 
|  | // |process_milestones|. The actual meanings of these bits are defined | 
|  | // (globally) by the caller(s). | 
|  | static void SetProcessMilestone(int milestone); | 
|  |  | 
|  | private: | 
|  | friend class TestAPI; | 
|  |  | 
|  | // SamplingThread is a separate thread used to suspend and sample stacks from | 
|  | // the target thread. | 
|  | class SamplingThread; | 
|  |  | 
|  | // Adds annotations to a Sample. | 
|  | static void RecordAnnotations(Sample* sample); | 
|  |  | 
|  | // This global variables holds the current system state and is recorded with | 
|  | // every captured sample, done on a separate thread which is why updates to | 
|  | // this must be atomic. A PostTask to move the the updates to that thread | 
|  | // would skew the timing and a lock could result in deadlock if the thread | 
|  | // making a change was also being profiled and got stopped. | 
|  | static subtle::Atomic32 process_milestones_; | 
|  |  | 
|  | // The thread whose stack will be sampled. | 
|  | PlatformThreadId thread_id_; | 
|  |  | 
|  | const SamplingParams params_; | 
|  |  | 
|  | const CompletedCallback completed_callback_; | 
|  |  | 
|  | // This starts "signaled", is reset when sampling begins, and is signaled | 
|  | // when that sampling is complete and the callback done. | 
|  | WaitableEvent profiling_inactive_; | 
|  |  | 
|  | // Object that does the native sampling. This is created during construction | 
|  | // and later passed to the sampling thread when profiling is started. | 
|  | std::unique_ptr<NativeStackSampler> native_sampler_; | 
|  |  | 
|  | // An ID uniquely identifying this profiler to the sampling thread. This | 
|  | // will be an internal "null" value when no collection has been started. | 
|  | int profiler_id_; | 
|  |  | 
|  | // Stored until it can be passed to the NativeStackSampler created in Start(). | 
|  | NativeStackSamplerTestDelegate* const test_delegate_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(StackSamplingProfiler); | 
|  | }; | 
|  |  | 
|  | // These operators permit types to be compared and used in a map of Samples, as | 
|  | // done in tests and by the metrics provider code. | 
|  | BASE_EXPORT bool operator==(const StackSamplingProfiler::Module& a, | 
|  | const StackSamplingProfiler::Module& b); | 
|  | BASE_EXPORT bool operator==(const StackSamplingProfiler::Sample& a, | 
|  | const StackSamplingProfiler::Sample& b); | 
|  | BASE_EXPORT bool operator!=(const StackSamplingProfiler::Sample& a, | 
|  | const StackSamplingProfiler::Sample& b); | 
|  | BASE_EXPORT bool operator<(const StackSamplingProfiler::Sample& a, | 
|  | const StackSamplingProfiler::Sample& b); | 
|  | BASE_EXPORT bool operator==(const StackSamplingProfiler::Frame& a, | 
|  | const StackSamplingProfiler::Frame& b); | 
|  | BASE_EXPORT bool operator<(const StackSamplingProfiler::Frame& a, | 
|  | const StackSamplingProfiler::Frame& b); | 
|  |  | 
|  | }  // namespace base | 
|  |  | 
|  | #endif  // BASE_PROFILER_STACK_SAMPLING_PROFILER_H_ |