blob: 254e269d41400b059de275e2b51d222c89e43376 [file] [log] [blame]
/*
* Copyright (C) 2010 Apple Inc. All rights reserved.
* Copyright (C) 2010 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. 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.
* 3. Neither the name of Apple Computer, Inc. ("Apple") 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 APPLE AND ITS 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 APPLE OR ITS 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.
*/
#include "config.h"
#include "core/inspector/InspectorProfilerAgent.h"
#include "bindings/core/v8/ScriptCallStackFactory.h"
#include "bindings/core/v8/V8Binding.h"
#include "core/frame/UseCounter.h"
#include "core/inspector/InjectedScript.h"
#include "core/inspector/InjectedScriptHost.h"
#include "core/inspector/InspectorState.h"
#include "core/inspector/InstrumentingAgents.h"
#include "core/inspector/ScriptCallStack.h"
#include <v8-profiler.h>
namespace blink {
namespace ProfilerAgentState {
static const char samplingInterval[] = "samplingInterval";
static const char userInitiatedProfiling[] = "userInitiatedProfiling";
static const char profilerEnabled[] = "profilerEnabled";
static const char nextProfileId[] = "nextProfileId";
}
namespace {
PassRefPtr<TypeBuilder::Array<TypeBuilder::Profiler::PositionTickInfo>> buildInspectorObjectForPositionTicks(const v8::CpuProfileNode* node)
{
RefPtr<TypeBuilder::Array<TypeBuilder::Profiler::PositionTickInfo>> array = TypeBuilder::Array<TypeBuilder::Profiler::PositionTickInfo>::create();
unsigned lineCount = node->GetHitLineCount();
if (!lineCount)
return array.release();
Vector<v8::CpuProfileNode::LineTick> entries(lineCount);
if (node->GetLineTicks(&entries[0], lineCount)) {
for (unsigned i = 0; i < lineCount; i++) {
RefPtr<TypeBuilder::Profiler::PositionTickInfo> line = TypeBuilder::Profiler::PositionTickInfo::create()
.setLine(entries[i].line)
.setTicks(entries[i].hit_count);
array->addItem(line);
}
}
return array.release();
}
PassRefPtr<TypeBuilder::Profiler::CPUProfileNode> buildInspectorObjectFor(const v8::CpuProfileNode* node)
{
v8::HandleScope handleScope(v8::Isolate::GetCurrent());
RefPtr<TypeBuilder::Array<TypeBuilder::Profiler::CPUProfileNode>> children = TypeBuilder::Array<TypeBuilder::Profiler::CPUProfileNode>::create();
const int childrenCount = node->GetChildrenCount();
for (int i = 0; i < childrenCount; i++) {
const v8::CpuProfileNode* child = node->GetChild(i);
children->addItem(buildInspectorObjectFor(child));
}
RefPtr<TypeBuilder::Array<TypeBuilder::Profiler::PositionTickInfo>> positionTicks = buildInspectorObjectForPositionTicks(node);
RefPtr<TypeBuilder::Profiler::CPUProfileNode> result = TypeBuilder::Profiler::CPUProfileNode::create()
.setFunctionName(toCoreString(node->GetFunctionName()))
.setScriptId(String::number(node->GetScriptId()))
.setUrl(toCoreString(node->GetScriptResourceName()))
.setLineNumber(node->GetLineNumber())
.setColumnNumber(node->GetColumnNumber())
.setHitCount(node->GetHitCount())
.setCallUID(node->GetCallUid())
.setChildren(children.release())
.setPositionTicks(positionTicks.release())
.setDeoptReason(node->GetBailoutReason())
.setId(node->GetNodeId());
return result.release();
}
PassRefPtr<TypeBuilder::Array<int>> buildInspectorObjectForSamples(v8::CpuProfile* v8profile)
{
RefPtr<TypeBuilder::Array<int>> array = TypeBuilder::Array<int>::create();
int count = v8profile->GetSamplesCount();
for (int i = 0; i < count; i++)
array->addItem(v8profile->GetSample(i)->GetNodeId());
return array.release();
}
PassRefPtr<TypeBuilder::Array<double>> buildInspectorObjectForTimestamps(v8::CpuProfile* v8profile)
{
RefPtr<TypeBuilder::Array<double>> array = TypeBuilder::Array<double>::create();
int count = v8profile->GetSamplesCount();
for (int i = 0; i < count; i++)
array->addItem(v8profile->GetSampleTimestamp(i));
return array.release();
}
PassRefPtr<TypeBuilder::Profiler::CPUProfile> createCPUProfile(v8::CpuProfile* v8profile)
{
RefPtr<TypeBuilder::Profiler::CPUProfile> profile = TypeBuilder::Profiler::CPUProfile::create()
.setHead(buildInspectorObjectFor(v8profile->GetTopDownRoot()))
.setStartTime(static_cast<double>(v8profile->GetStartTime()) / 1000000)
.setEndTime(static_cast<double>(v8profile->GetEndTime()) / 1000000);
profile->setSamples(buildInspectorObjectForSamples(v8profile));
profile->setTimestamps(buildInspectorObjectForTimestamps(v8profile));
return profile.release();
}
PassRefPtr<TypeBuilder::Debugger::Location> currentDebugLocation()
{
RefPtrWillBeRawPtr<ScriptCallStack> callStack(currentScriptCallStack(1));
const ScriptCallFrame& lastCaller = callStack->at(0);
RefPtr<TypeBuilder::Debugger::Location> location = TypeBuilder::Debugger::Location::create()
.setScriptId(lastCaller.scriptId())
.setLineNumber(lastCaller.lineNumber());
location->setColumnNumber(lastCaller.columnNumber());
return location.release();
}
} // namespace
class InspectorProfilerAgent::ProfileDescriptor {
public:
ProfileDescriptor(const String& id, const String& title)
: m_id(id)
, m_title(title) { }
String m_id;
String m_title;
};
PassOwnPtrWillBeRawPtr<InspectorProfilerAgent> InspectorProfilerAgent::create(v8::Isolate* isolate, InjectedScriptManager* injectedScriptManager, Client* client)
{
return adoptPtrWillBeNoop(new InspectorProfilerAgent(isolate, injectedScriptManager, client));
}
InspectorProfilerAgent::InspectorProfilerAgent(v8::Isolate* isolate, InjectedScriptManager* injectedScriptManager, Client* client)
: InspectorBaseAgent<InspectorProfilerAgent, InspectorFrontend::Profiler>("Profiler")
, m_isolate(isolate)
, m_injectedScriptManager(injectedScriptManager)
, m_recordingCPUProfile(false)
, m_client(client)
{
}
InspectorProfilerAgent::~InspectorProfilerAgent()
{
}
void InspectorProfilerAgent::consoleProfile(ExecutionContext* context, const String& title)
{
UseCounter::count(context, UseCounter::DevToolsConsoleProfile);
ASSERT(frontend() && enabled());
String id = nextProfileId();
m_startedProfiles.append(ProfileDescriptor(id, title));
startProfiling(id);
frontend()->consoleProfileStarted(id, currentDebugLocation(), title.isNull() ? 0 : &title);
}
void InspectorProfilerAgent::consoleProfileEnd(const String& title)
{
ASSERT(frontend() && enabled());
String id;
String resolvedTitle;
// Take last started profile if no title was passed.
if (title.isNull()) {
if (m_startedProfiles.isEmpty())
return;
id = m_startedProfiles.last().m_id;
resolvedTitle = m_startedProfiles.last().m_title;
m_startedProfiles.removeLast();
} else {
for (size_t i = 0; i < m_startedProfiles.size(); i++) {
if (m_startedProfiles[i].m_title == title) {
resolvedTitle = title;
id = m_startedProfiles[i].m_id;
m_startedProfiles.remove(i);
break;
}
}
if (id.isEmpty())
return;
}
RefPtr<TypeBuilder::Profiler::CPUProfile> profile = stopProfiling(id, true);
if (!profile)
return;
RefPtr<TypeBuilder::Debugger::Location> location = currentDebugLocation();
frontend()->consoleProfileFinished(id, location, profile, resolvedTitle.isNull() ? 0 : &resolvedTitle);
}
void InspectorProfilerAgent::enable(ErrorString*)
{
m_state->setBoolean(ProfilerAgentState::profilerEnabled, true);
doEnable();
}
void InspectorProfilerAgent::doEnable()
{
m_instrumentingAgents->setInspectorProfilerAgent(this);
}
void InspectorProfilerAgent::disable(ErrorString*)
{
for (Vector<ProfileDescriptor>::reverse_iterator it = m_startedProfiles.rbegin(); it != m_startedProfiles.rend(); ++it)
stopProfiling(it->m_id, false);
m_startedProfiles.clear();
stop(0, 0);
m_instrumentingAgents->setInspectorProfilerAgent(nullptr);
m_state->setBoolean(ProfilerAgentState::profilerEnabled, false);
}
bool InspectorProfilerAgent::enabled()
{
return m_state->getBoolean(ProfilerAgentState::profilerEnabled);
}
void InspectorProfilerAgent::setSamplingInterval(ErrorString* error, int interval)
{
if (m_recordingCPUProfile) {
*error = "Cannot change sampling interval when profiling.";
return;
}
m_state->setLong(ProfilerAgentState::samplingInterval, interval);
m_isolate->GetCpuProfiler()->SetSamplingInterval(interval);
}
void InspectorProfilerAgent::restore()
{
if (m_state->getBoolean(ProfilerAgentState::profilerEnabled))
doEnable();
if (long interval = m_state->getLong(ProfilerAgentState::samplingInterval, 0))
m_isolate->GetCpuProfiler()->SetSamplingInterval(interval);
if (m_state->getBoolean(ProfilerAgentState::userInitiatedProfiling)) {
ErrorString error;
start(&error);
}
}
void InspectorProfilerAgent::start(ErrorString* error)
{
if (m_recordingCPUProfile)
return;
if (!enabled()) {
*error = "Profiler is not enabled";
return;
}
m_recordingCPUProfile = true;
if (m_client)
m_client->profilingStarted();
m_frontendInitiatedProfileId = nextProfileId();
startProfiling(m_frontendInitiatedProfileId);
m_state->setBoolean(ProfilerAgentState::userInitiatedProfiling, true);
}
void InspectorProfilerAgent::stop(ErrorString* errorString, RefPtr<TypeBuilder::Profiler::CPUProfile>& profile)
{
stop(errorString, &profile);
}
void InspectorProfilerAgent::stop(ErrorString* errorString, RefPtr<TypeBuilder::Profiler::CPUProfile>* profile)
{
if (!m_recordingCPUProfile) {
if (errorString)
*errorString = "No recording profiles found";
return;
}
m_recordingCPUProfile = false;
if (m_client)
m_client->profilingStopped();
RefPtr<TypeBuilder::Profiler::CPUProfile> cpuProfile = stopProfiling(m_frontendInitiatedProfileId, !!profile);
if (profile) {
*profile = cpuProfile;
if (!cpuProfile && errorString)
*errorString = "Profile wasn't found";
}
m_frontendInitiatedProfileId = String();
m_state->setBoolean(ProfilerAgentState::userInitiatedProfiling, false);
}
String InspectorProfilerAgent::nextProfileId()
{
long nextId = m_state->getLong(ProfilerAgentState::nextProfileId, 1);
m_state->setLong(ProfilerAgentState::nextProfileId, nextId + 1);
return String::number(nextId);
}
void InspectorProfilerAgent::startProfiling(const String& title)
{
v8::HandleScope handleScope(m_isolate);
m_isolate->GetCpuProfiler()->StartProfiling(v8String(m_isolate, title), true);
}
PassRefPtr<TypeBuilder::Profiler::CPUProfile> InspectorProfilerAgent::stopProfiling(const String& title, bool serialize)
{
v8::HandleScope handleScope(m_isolate);
v8::CpuProfile* profile = m_isolate->GetCpuProfiler()->StopProfiling(v8String(m_isolate, title));
if (!profile)
return nullptr;
RefPtr<TypeBuilder::Profiler::CPUProfile> result;
if (serialize)
result = createCPUProfile(profile);
profile->Delete();
return result.release();
}
bool InspectorProfilerAgent::isRecording() const
{
return m_recordingCPUProfile || !m_startedProfiles.isEmpty();
}
void InspectorProfilerAgent::idleFinished()
{
if (!isRecording())
return;
m_isolate->GetCpuProfiler()->SetIdle(false);
}
void InspectorProfilerAgent::idleStarted()
{
if (!isRecording())
return;
m_isolate->GetCpuProfiler()->SetIdle(true);
}
void InspectorProfilerAgent::willProcessTask()
{
idleFinished();
}
void InspectorProfilerAgent::didProcessTask()
{
idleStarted();
}
void InspectorProfilerAgent::willEnterNestedRunLoop()
{
idleStarted();
}
void InspectorProfilerAgent::didLeaveNestedRunLoop()
{
idleFinished();
}
DEFINE_TRACE(InspectorProfilerAgent)
{
visitor->trace(m_injectedScriptManager);
InspectorBaseAgent::trace(visitor);
}
} // namespace blink