2013-04-15 13:57:41 +00:00
|
|
|
// Copyright 2013 the V8 project authors. All rights reserved.
|
2014-04-29 06:42:26 +00:00
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
2013-04-15 13:57:41 +00:00
|
|
|
|
|
|
|
#ifndef V8_SAMPLER_H_
|
|
|
|
#define V8_SAMPLER_H_
|
|
|
|
|
2014-06-05 12:14:47 +00:00
|
|
|
#include "src/base/atomicops.h"
|
2014-06-03 08:12:43 +00:00
|
|
|
#include "src/frames.h"
|
|
|
|
#include "src/globals.h"
|
2013-04-15 13:57:41 +00:00
|
|
|
|
|
|
|
namespace v8 {
|
|
|
|
namespace internal {
|
|
|
|
|
|
|
|
class Isolate;
|
|
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Sampler
|
|
|
|
//
|
|
|
|
// A sampler periodically samples the state of the VM and optionally
|
|
|
|
// (if used for profiling) the program counter and stack pointer for
|
|
|
|
// the thread that created it.
|
|
|
|
|
2013-07-03 16:20:59 +00:00
|
|
|
struct RegisterState {
|
|
|
|
RegisterState() : pc(NULL), sp(NULL), fp(NULL) {}
|
|
|
|
Address pc; // Instruction pointer.
|
|
|
|
Address sp; // Stack pointer.
|
|
|
|
Address fp; // Frame pointer.
|
|
|
|
};
|
|
|
|
|
2013-04-15 13:57:41 +00:00
|
|
|
// TickSample captures the information collected for each sample.
|
|
|
|
struct TickSample {
|
|
|
|
TickSample()
|
|
|
|
: state(OTHER),
|
|
|
|
pc(NULL),
|
|
|
|
external_callback(NULL),
|
2013-04-19 11:55:01 +00:00
|
|
|
frames_count(0),
|
2013-07-03 14:04:37 +00:00
|
|
|
has_external_callback(false),
|
|
|
|
top_frame_type(StackFrame::NONE) {}
|
2013-07-03 16:20:59 +00:00
|
|
|
void Init(Isolate* isolate, const RegisterState& state);
|
2013-04-15 13:57:41 +00:00
|
|
|
StateTag state; // The state of the VM.
|
|
|
|
Address pc; // Instruction pointer.
|
2013-04-19 11:55:01 +00:00
|
|
|
union {
|
|
|
|
Address tos; // Top stack value (*sp).
|
|
|
|
Address external_callback;
|
|
|
|
};
|
2014-06-24 16:00:51 +00:00
|
|
|
static const unsigned kMaxFramesCountLog2 = 8;
|
|
|
|
static const unsigned kMaxFramesCount = (1 << kMaxFramesCountLog2) - 1;
|
2013-04-15 13:57:41 +00:00
|
|
|
Address stack[kMaxFramesCount]; // Call stack.
|
2014-06-30 13:25:46 +00:00
|
|
|
base::TimeTicks timestamp;
|
2014-06-24 16:00:51 +00:00
|
|
|
unsigned frames_count : kMaxFramesCountLog2; // Number of captured frames.
|
2013-04-19 11:55:01 +00:00
|
|
|
bool has_external_callback : 1;
|
2013-07-03 14:04:37 +00:00
|
|
|
StackFrame::Type top_frame_type : 4;
|
2013-04-15 13:57:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
class Sampler {
|
|
|
|
public:
|
|
|
|
// Initializes the Sampler support. Called once at VM startup.
|
|
|
|
static void SetUp();
|
|
|
|
static void TearDown();
|
|
|
|
|
|
|
|
// Initialize sampler.
|
|
|
|
Sampler(Isolate* isolate, int interval);
|
|
|
|
virtual ~Sampler();
|
|
|
|
|
|
|
|
Isolate* isolate() const { return isolate_; }
|
|
|
|
int interval() const { return interval_; }
|
|
|
|
|
|
|
|
// Performs stack sampling.
|
2013-07-03 16:20:59 +00:00
|
|
|
void SampleStack(const RegisterState& regs);
|
2013-04-15 13:57:41 +00:00
|
|
|
|
|
|
|
// Start and stop sampler.
|
|
|
|
void Start();
|
|
|
|
void Stop();
|
|
|
|
|
2013-08-26 07:17:12 +00:00
|
|
|
// Whether the sampling thread should use this Sampler for CPU profiling?
|
|
|
|
bool IsProfiling() const {
|
2014-06-05 12:14:47 +00:00
|
|
|
return base::NoBarrier_Load(&profiling_) > 0 &&
|
|
|
|
!base::NoBarrier_Load(&has_processing_thread_);
|
2013-08-26 07:17:12 +00:00
|
|
|
}
|
2013-08-29 14:03:38 +00:00
|
|
|
void IncreaseProfilingDepth();
|
|
|
|
void DecreaseProfilingDepth();
|
2013-04-15 13:57:41 +00:00
|
|
|
|
|
|
|
// Whether the sampler is running (that is, consumes resources).
|
2014-06-05 12:14:47 +00:00
|
|
|
bool IsActive() const { return base::NoBarrier_Load(&active_); }
|
2013-04-15 13:57:41 +00:00
|
|
|
|
2013-08-29 14:03:38 +00:00
|
|
|
void DoSample();
|
2013-08-26 07:17:12 +00:00
|
|
|
// If true next sample must be initiated on the profiler event processor
|
|
|
|
// thread right after latest sample is processed.
|
|
|
|
void SetHasProcessingThread(bool value) {
|
2014-06-05 12:14:47 +00:00
|
|
|
base::NoBarrier_Store(&has_processing_thread_, value);
|
2013-08-26 07:17:12 +00:00
|
|
|
}
|
|
|
|
|
2013-04-15 13:57:41 +00:00
|
|
|
// Used in tests to make sure that stack sampling is performed.
|
2013-07-10 12:56:58 +00:00
|
|
|
unsigned js_and_external_sample_count() const {
|
|
|
|
return js_and_external_sample_count_;
|
|
|
|
}
|
|
|
|
void StartCountingSamples() {
|
|
|
|
is_counting_samples_ = true;
|
|
|
|
js_and_external_sample_count_ = 0;
|
|
|
|
}
|
2013-04-15 13:57:41 +00:00
|
|
|
|
|
|
|
class PlatformData;
|
|
|
|
PlatformData* platform_data() const { return data_; }
|
|
|
|
|
2013-07-03 16:20:59 +00:00
|
|
|
protected:
|
|
|
|
// This method is called for each sampling period with the current
|
|
|
|
// program counter.
|
|
|
|
virtual void Tick(TickSample* sample) = 0;
|
|
|
|
|
2013-04-15 13:57:41 +00:00
|
|
|
private:
|
2014-06-05 12:14:47 +00:00
|
|
|
void SetActive(bool value) { base::NoBarrier_Store(&active_, value); }
|
2013-04-15 13:57:41 +00:00
|
|
|
|
|
|
|
Isolate* isolate_;
|
|
|
|
const int interval_;
|
2014-06-05 12:14:47 +00:00
|
|
|
base::Atomic32 profiling_;
|
|
|
|
base::Atomic32 has_processing_thread_;
|
|
|
|
base::Atomic32 active_;
|
2013-04-15 13:57:41 +00:00
|
|
|
PlatformData* data_; // Platform specific data.
|
2013-07-10 12:56:58 +00:00
|
|
|
bool is_counting_samples_;
|
|
|
|
// Counts stack samples taken in JS VM state.
|
|
|
|
unsigned js_and_external_sample_count_;
|
2013-04-15 13:57:41 +00:00
|
|
|
DISALLOW_IMPLICIT_CONSTRUCTORS(Sampler);
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
} } // namespace v8::internal
|
|
|
|
|
|
|
|
#endif // V8_SAMPLER_H_
|