69c2f54d32
Stack iterator takes return address based on the frame pointer (ebp) and detects JS frames based on value at fp + StandardFrameConstants::kMarkerOffset. So in order the iterator to work correctly this values should be already setup for the current function. Stack frame is constructed at the very beginning of JS function code and destroyed before return. If sample is taken before before the frame construction is completed or after it was destroyed the stack iterator will wrongly think that FP points at the current functions frame base and will skip callers frame. To avoid this we mark code ranges where stack frame doesn't exist and completely ignore such samples. This fixes cctest/test-cpu-profiler/CollectCpuProfile flakiness. BUG=v8:2628 R=jkummerow@chromium.org Review URL: https://codereview.chromium.org/14253015 git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14670 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
276 lines
9.4 KiB
C++
276 lines
9.4 KiB
C++
// Copyright 2012 the V8 project authors. All rights reserved.
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are
|
|
// met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the following
|
|
// disclaimer in the documentation and/or other materials provided
|
|
// with the distribution.
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
// contributors may be used to endorse or promote products derived
|
|
// from this software without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
#ifndef V8_CPU_PROFILER_H_
|
|
#define V8_CPU_PROFILER_H_
|
|
|
|
#include "allocation.h"
|
|
#include "atomicops.h"
|
|
#include "circular-queue.h"
|
|
#include "sampler.h"
|
|
#include "unbound-queue.h"
|
|
|
|
namespace v8 {
|
|
namespace internal {
|
|
|
|
// Forward declarations.
|
|
class CodeEntry;
|
|
class CodeMap;
|
|
class CompilationInfo;
|
|
class CpuProfile;
|
|
class CpuProfilesCollection;
|
|
class ProfileGenerator;
|
|
class TokenEnumerator;
|
|
|
|
#define CODE_EVENTS_TYPE_LIST(V) \
|
|
V(CODE_CREATION, CodeCreateEventRecord) \
|
|
V(CODE_MOVE, CodeMoveEventRecord) \
|
|
V(SHARED_FUNC_MOVE, SharedFunctionInfoMoveEventRecord)
|
|
|
|
|
|
class CodeEventRecord {
|
|
public:
|
|
#define DECLARE_TYPE(type, ignore) type,
|
|
enum Type {
|
|
NONE = 0,
|
|
CODE_EVENTS_TYPE_LIST(DECLARE_TYPE)
|
|
NUMBER_OF_TYPES
|
|
};
|
|
#undef DECLARE_TYPE
|
|
|
|
Type type;
|
|
unsigned order;
|
|
};
|
|
|
|
|
|
class CodeCreateEventRecord : public CodeEventRecord {
|
|
public:
|
|
Address start;
|
|
CodeEntry* entry;
|
|
unsigned size;
|
|
Address shared;
|
|
|
|
INLINE(void UpdateCodeMap(CodeMap* code_map));
|
|
};
|
|
|
|
|
|
class CodeMoveEventRecord : public CodeEventRecord {
|
|
public:
|
|
Address from;
|
|
Address to;
|
|
|
|
INLINE(void UpdateCodeMap(CodeMap* code_map));
|
|
};
|
|
|
|
|
|
class SharedFunctionInfoMoveEventRecord : public CodeEventRecord {
|
|
public:
|
|
Address from;
|
|
Address to;
|
|
|
|
INLINE(void UpdateCodeMap(CodeMap* code_map));
|
|
};
|
|
|
|
|
|
class TickSampleEventRecord {
|
|
public:
|
|
// The parameterless constructor is used when we dequeue data from
|
|
// the ticks buffer.
|
|
TickSampleEventRecord() { }
|
|
explicit TickSampleEventRecord(unsigned order)
|
|
: filler(1),
|
|
order(order) {
|
|
ASSERT(filler != SamplingCircularQueue::kClear);
|
|
}
|
|
|
|
// The first machine word of a TickSampleEventRecord must not ever
|
|
// become equal to SamplingCircularQueue::kClear. As both order and
|
|
// TickSample's first field are not reliable in this sense (order
|
|
// can overflow, TickSample can have all fields reset), we are
|
|
// forced to use an artificial filler field.
|
|
int filler;
|
|
unsigned order;
|
|
TickSample sample;
|
|
|
|
static TickSampleEventRecord* cast(void* value) {
|
|
return reinterpret_cast<TickSampleEventRecord*>(value);
|
|
}
|
|
};
|
|
|
|
|
|
// This class implements both the profile events processor thread and
|
|
// methods called by event producers: VM and stack sampler threads.
|
|
class ProfilerEventsProcessor : public Thread {
|
|
public:
|
|
ProfilerEventsProcessor(ProfileGenerator* generator,
|
|
CpuProfilesCollection* profiles);
|
|
virtual ~ProfilerEventsProcessor() {}
|
|
|
|
// Thread control.
|
|
virtual void Run();
|
|
inline void Stop() { running_ = false; }
|
|
INLINE(bool running()) { return running_; }
|
|
|
|
// Events adding methods. Called by VM threads.
|
|
void CallbackCreateEvent(Logger::LogEventsAndTags tag,
|
|
const char* prefix, Name* name,
|
|
Address start);
|
|
void CodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
Name* name,
|
|
String* resource_name, int line_number,
|
|
Address start, unsigned size,
|
|
Address shared,
|
|
CompilationInfo* info);
|
|
void CodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
const char* name,
|
|
Address start, unsigned size);
|
|
void CodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
int args_count,
|
|
Address start, unsigned size);
|
|
void CodeMoveEvent(Address from, Address to);
|
|
void CodeDeleteEvent(Address from);
|
|
void SharedFunctionInfoMoveEvent(Address from, Address to);
|
|
void RegExpCodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
const char* prefix, String* name,
|
|
Address start, unsigned size);
|
|
// Puts current stack into tick sample events buffer.
|
|
void AddCurrentStack();
|
|
|
|
// Tick sample events are filled directly in the buffer of the circular
|
|
// queue (because the structure is of fixed width, but usually not all
|
|
// stack frame entries are filled.) This method returns a pointer to the
|
|
// next record of the buffer.
|
|
INLINE(TickSample* TickSampleEvent());
|
|
|
|
private:
|
|
union CodeEventsContainer {
|
|
CodeEventRecord generic;
|
|
#define DECLARE_CLASS(ignore, type) type type##_;
|
|
CODE_EVENTS_TYPE_LIST(DECLARE_CLASS)
|
|
#undef DECLARE_TYPE
|
|
};
|
|
|
|
// Called from events processing thread (Run() method.)
|
|
bool ProcessCodeEvent(unsigned* dequeue_order);
|
|
bool ProcessTicks(unsigned dequeue_order);
|
|
|
|
INLINE(static bool FilterOutCodeCreateEvent(Logger::LogEventsAndTags tag));
|
|
|
|
ProfileGenerator* generator_;
|
|
CpuProfilesCollection* profiles_;
|
|
bool running_;
|
|
UnboundQueue<CodeEventsContainer> events_buffer_;
|
|
SamplingCircularQueue ticks_buffer_;
|
|
UnboundQueue<TickSampleEventRecord> ticks_from_vm_buffer_;
|
|
unsigned enqueue_order_;
|
|
};
|
|
|
|
|
|
#define PROFILE(IsolateGetter, Call) \
|
|
do { \
|
|
Isolate* cpu_profiler_isolate = (IsolateGetter); \
|
|
LOG_CODE_EVENT(cpu_profiler_isolate, Call); \
|
|
CpuProfiler* cpu_profiler = cpu_profiler_isolate->cpu_profiler(); \
|
|
if (cpu_profiler->is_profiling()) { \
|
|
cpu_profiler->Call; \
|
|
} \
|
|
} while (false)
|
|
|
|
|
|
class CpuProfiler {
|
|
public:
|
|
explicit CpuProfiler(Isolate* isolate);
|
|
~CpuProfiler();
|
|
|
|
void StartProfiling(const char* title, bool record_samples = false);
|
|
void StartProfiling(String* title, bool record_samples);
|
|
CpuProfile* StopProfiling(const char* title);
|
|
CpuProfile* StopProfiling(Object* security_token, String* title);
|
|
int GetProfilesCount();
|
|
CpuProfile* GetProfile(Object* security_token, int index);
|
|
CpuProfile* FindProfile(Object* security_token, unsigned uid);
|
|
void DeleteAllProfiles();
|
|
void DeleteProfile(CpuProfile* profile);
|
|
bool HasDetachedProfiles();
|
|
|
|
// Invoked from stack sampler (thread or signal handler.)
|
|
TickSample* TickSampleEvent();
|
|
|
|
// Must be called via PROFILE macro, otherwise will crash when
|
|
// profiling is not enabled.
|
|
void CallbackEvent(Name* name, Address entry_point);
|
|
void CodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
Code* code, const char* comment);
|
|
void CodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
Code* code, Name* name);
|
|
void CodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
Code* code,
|
|
SharedFunctionInfo* shared,
|
|
CompilationInfo* info,
|
|
Name* name);
|
|
void CodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
Code* code,
|
|
SharedFunctionInfo* shared,
|
|
CompilationInfo* info,
|
|
String* source, int line);
|
|
void CodeCreateEvent(Logger::LogEventsAndTags tag,
|
|
Code* code, int args_count);
|
|
void CodeMovingGCEvent() {}
|
|
void CodeMoveEvent(Address from, Address to);
|
|
void CodeDeleteEvent(Address from);
|
|
void GetterCallbackEvent(Name* name, Address entry_point);
|
|
void RegExpCodeCreateEvent(Code* code, String* source);
|
|
void SetterCallbackEvent(Name* name, Address entry_point);
|
|
void SharedFunctionInfoMoveEvent(Address from, Address to);
|
|
|
|
INLINE(bool is_profiling() const) { return is_profiling_; }
|
|
|
|
private:
|
|
void StartProcessorIfNotStarted();
|
|
void StopProcessorIfLastProfile(const char* title);
|
|
void StopProcessor();
|
|
void ResetProfiles();
|
|
|
|
Isolate* isolate_;
|
|
CpuProfilesCollection* profiles_;
|
|
unsigned next_profile_uid_;
|
|
TokenEnumerator* token_enumerator_;
|
|
ProfileGenerator* generator_;
|
|
ProfilerEventsProcessor* processor_;
|
|
int saved_logging_nesting_;
|
|
bool need_to_stop_sampler_;
|
|
bool is_profiling_;
|
|
|
|
private:
|
|
DISALLOW_COPY_AND_ASSIGN(CpuProfiler);
|
|
};
|
|
|
|
} } // namespace v8::internal
|
|
|
|
|
|
#endif // V8_CPU_PROFILER_H_
|