blob: 64c60c4237197549172650c30c0a8a36cd597972 [file] [log] [blame]
// Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
#ifndef VM_PROFILER_H_
#define VM_PROFILER_H_
#include "vm/allocation.h"
#include "vm/code_observers.h"
#include "vm/globals.h"
#include "vm/tags.h"
#include "vm/thread.h"
#include "vm/thread_interrupter.h"
namespace dart {
// Forward declarations.
class JSONArray;
class JSONStream;
class ProfilerCodeRegionTable;
class Sample;
class SampleBuffer;
// Profiler
class Profiler : public AllStatic {
public:
static void InitOnce();
static void Shutdown();
static void SetSampleDepth(intptr_t depth);
static void SetSamplePeriod(intptr_t period);
static void InitProfilingForIsolate(Isolate* isolate,
bool shared_buffer = true);
static void ShutdownProfilingForIsolate(Isolate* isolate);
static void BeginExecution(Isolate* isolate);
static void EndExecution(Isolate* isolate);
static void PrintToJSONStream(Isolate* isolate, JSONStream* stream,
bool full, bool use_tags);
static void WriteProfile(Isolate* isolate);
static SampleBuffer* sample_buffer() {
return sample_buffer_;
}
private:
static bool initialized_;
static Monitor* monitor_;
static void RecordSampleInterruptCallback(const InterruptedThreadState& state,
void* data);
static SampleBuffer* sample_buffer_;
};
class IsolateProfilerData {
public:
IsolateProfilerData(SampleBuffer* sample_buffer, bool own_sample_buffer);
~IsolateProfilerData();
SampleBuffer* sample_buffer() const { return sample_buffer_; }
void set_sample_buffer(SampleBuffer* sample_buffer) {
sample_buffer_ = sample_buffer;
}
private:
SampleBuffer* sample_buffer_;
bool own_sample_buffer_;
DISALLOW_COPY_AND_ASSIGN(IsolateProfilerData);
};
class SampleVisitor {
public:
explicit SampleVisitor(Isolate* isolate) : isolate_(isolate), visited_(0) { }
virtual ~SampleVisitor() {}
virtual void VisitSample(Sample* sample) = 0;
intptr_t visited() const {
return visited_;
}
void IncrementVisited() {
visited_++;
}
Isolate* isolate() const {
return isolate_;
}
private:
Isolate* isolate_;
intptr_t visited_;
DISALLOW_IMPLICIT_CONSTRUCTORS(SampleVisitor);
};
// The maximum number of stack frames a sample can hold.
#define kSampleFramesSize 256
// Each Sample holds a stack trace from an isolate.
class Sample {
public:
void Init(Isolate* isolate, int64_t timestamp, ThreadId tid) {
timestamp_ = timestamp;
tid_ = tid;
isolate_ = isolate;
vm_tag_ = VMTag::kInvalidTagId;
for (intptr_t i = 0; i < kSampleFramesSize; i++) {
pcs_[i] = 0;
}
}
// Isolate sample was taken from.
Isolate* isolate() const {
return isolate_;
}
// Timestamp sample was taken at.
int64_t timestamp() const {
return timestamp_;
}
// Get stack trace entry.
uword At(intptr_t i) const {
ASSERT(i >= 0);
ASSERT(i < kSampleFramesSize);
return pcs_[i];
}
// Set stack trace entry.
void SetAt(intptr_t i, uword pc) {
ASSERT(i >= 0);
ASSERT(i < kSampleFramesSize);
pcs_[i] = pc;
}
uword vm_tag() const {
return vm_tag_;
}
void set_vm_tag(uword tag) {
ASSERT(tag != VMTag::kInvalidTagId);
vm_tag_ = tag;
}
private:
int64_t timestamp_;
ThreadId tid_;
Isolate* isolate_;
uword vm_tag_;
uword pcs_[kSampleFramesSize];
};
// Ring buffer of Samples that is (usually) shared by many isolates.
class SampleBuffer {
public:
static const intptr_t kDefaultBufferCapacity = 120000; // 2 minutes @ 1000hz.
explicit SampleBuffer(intptr_t capacity = kDefaultBufferCapacity) {
samples_ = reinterpret_cast<Sample*>(calloc(capacity, sizeof(*samples_)));
capacity_ = capacity;
cursor_ = 0;
}
~SampleBuffer() {
if (samples_ != NULL) {
free(samples_);
samples_ = NULL;
cursor_ = 0;
capacity_ = 0;
}
}
intptr_t capacity() const { return capacity_; }
Sample* ReserveSample();
Sample* At(intptr_t idx) const {
ASSERT(idx >= 0);
ASSERT(idx < capacity_);
return &samples_[idx];
}
void VisitSamples(SampleVisitor* visitor) {
ASSERT(visitor != NULL);
Sample sample;
const intptr_t length = capacity();
for (intptr_t i = 0; i < length; i++) {
// Copy the sample.
sample = *At(i);
if (sample.isolate() != visitor->isolate()) {
// Another isolate.
continue;
}
if (sample.timestamp() == 0) {
// Empty.
continue;
}
if (sample.At(0) == 0) {
// No frames.
continue;
}
visitor->IncrementVisited();
visitor->VisitSample(&sample);
}
}
private:
Sample* samples_;
intptr_t capacity_;
uintptr_t cursor_;
DISALLOW_COPY_AND_ASSIGN(SampleBuffer);
};
} // namespace dart
#endif // VM_PROFILER_H_