Commit b0e9116d authored by rmcilroy's avatar rmcilroy Committed by Commit bot

[Compiler] Track Ignition background compilation separately in RuntimeStats.

Tracks background compilation of Ignition in a separate bucket from main thread
compilation. Also add some more compilation buckets for functions which can take a
significant proportion of compilation.

BUG=v8:5203,v8:5215

Review-Url: https://codereview.chromium.org/2577263002
Cr-Commit-Position: refs/heads/master@{#42026}
parent b8f93dfe
...@@ -98,6 +98,8 @@ CompilationJob::Status CompilationJob::ExecuteJob() { ...@@ -98,6 +98,8 @@ CompilationJob::Status CompilationJob::ExecuteJob() {
no_handles.reset(new DisallowHandleAllocation()); no_handles.reset(new DisallowHandleAllocation());
no_deref.reset(new DisallowHandleDereference()); no_deref.reset(new DisallowHandleDereference());
no_dependency_change.reset(new DisallowCodeDependencyChange()); no_dependency_change.reset(new DisallowCodeDependencyChange());
executed_on_background_thread_ =
!ThreadId::Current().Equals(info()->isolate()->thread_id());
} else { } else {
DCHECK(ThreadId::Current().Equals(info()->isolate()->thread_id())); DCHECK(ThreadId::Current().Equals(info()->isolate()->thread_id()));
} }
...@@ -477,6 +479,8 @@ void EnsureSharedFunctionInfosArrayOnScript(ParseInfo* info) { ...@@ -477,6 +479,8 @@ void EnsureSharedFunctionInfosArrayOnScript(ParseInfo* info) {
} }
MUST_USE_RESULT MaybeHandle<Code> GetUnoptimizedCode(CompilationInfo* info) { MUST_USE_RESULT MaybeHandle<Code> GetUnoptimizedCode(CompilationInfo* info) {
RuntimeCallTimerScope runtimeTimer(
info->isolate(), &RuntimeCallStats::CompileGetUnoptimizedCode);
VMState<COMPILER> state(info->isolate()); VMState<COMPILER> state(info->isolate());
PostponeInterruptsScope postpone(info->isolate()); PostponeInterruptsScope postpone(info->isolate());
...@@ -499,6 +503,9 @@ MUST_USE_RESULT MaybeHandle<Code> GetUnoptimizedCode(CompilationInfo* info) { ...@@ -499,6 +503,9 @@ MUST_USE_RESULT MaybeHandle<Code> GetUnoptimizedCode(CompilationInfo* info) {
MUST_USE_RESULT MaybeHandle<Code> GetCodeFromOptimizedCodeMap( MUST_USE_RESULT MaybeHandle<Code> GetCodeFromOptimizedCodeMap(
Handle<JSFunction> function, BailoutId osr_ast_id) { Handle<JSFunction> function, BailoutId osr_ast_id) {
RuntimeCallTimerScope runtimeTimer(
function->GetIsolate(),
&RuntimeCallStats::CompileGetFromOptimizedCodeMap);
Handle<SharedFunctionInfo> shared(function->shared()); Handle<SharedFunctionInfo> shared(function->shared());
DisallowHeapAllocation no_gc; DisallowHeapAllocation no_gc;
CodeAndLiterals cached = shared->SearchOptimizedCodeMap( CodeAndLiterals cached = shared->SearchOptimizedCodeMap(
...@@ -882,8 +889,6 @@ MaybeHandle<Code> GetLazyCode(Handle<JSFunction> function) { ...@@ -882,8 +889,6 @@ MaybeHandle<Code> GetLazyCode(Handle<JSFunction> function) {
DCHECK(!isolate->has_pending_exception()); DCHECK(!isolate->has_pending_exception());
DCHECK(!function->is_compiled()); DCHECK(!function->is_compiled());
TimerEventScope<TimerEventCompileCode> compile_timer(isolate); TimerEventScope<TimerEventCompileCode> compile_timer(isolate);
RuntimeCallTimerScope runtimeTimer(isolate,
&RuntimeCallStats::CompileCodeLazy);
TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("v8.compile"), "V8.CompileCode"); TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("v8.compile"), "V8.CompileCode");
AggregatedHistogramTimerScope timer(isolate->counters()->compile_lazy()); AggregatedHistogramTimerScope timer(isolate->counters()->compile_lazy());
...@@ -1045,6 +1050,8 @@ Handle<SharedFunctionInfo> CompileToplevel(CompilationInfo* info) { ...@@ -1045,6 +1050,8 @@ Handle<SharedFunctionInfo> CompileToplevel(CompilationInfo* info) {
bool Compiler::Analyze(ParseInfo* info) { bool Compiler::Analyze(ParseInfo* info) {
DCHECK_NOT_NULL(info->literal()); DCHECK_NOT_NULL(info->literal());
RuntimeCallTimerScope runtimeTimer(info->isolate(),
&RuntimeCallStats::CompileAnalyse);
if (!Rewriter::Rewrite(info)) return false; if (!Rewriter::Rewrite(info)) return false;
DeclarationScope::Analyze(info, AnalyzeMode::kRegular); DeclarationScope::Analyze(info, AnalyzeMode::kRegular);
if (!Renumber(info)) return false; if (!Renumber(info)) return false;
......
...@@ -167,7 +167,8 @@ class CompilationJob { ...@@ -167,7 +167,8 @@ class CompilationJob {
: info_(info), : info_(info),
compiler_name_(compiler_name), compiler_name_(compiler_name),
state_(initial_state), state_(initial_state),
stack_limit_(isolate->stack_guard()->real_climit()) {} stack_limit_(isolate->stack_guard()->real_climit()),
executed_on_background_thread_(false) {}
virtual ~CompilationJob() {} virtual ~CompilationJob() {}
// Prepare the compile job. Must be called on the main thread. // Prepare the compile job. Must be called on the main thread.
...@@ -196,6 +197,11 @@ class CompilationJob { ...@@ -196,6 +197,11 @@ class CompilationJob {
void set_stack_limit(uintptr_t stack_limit) { stack_limit_ = stack_limit; } void set_stack_limit(uintptr_t stack_limit) { stack_limit_ = stack_limit; }
uintptr_t stack_limit() const { return stack_limit_; } uintptr_t stack_limit() const { return stack_limit_; }
bool executed_on_background_thread() const {
DCHECK_IMPLIES(!can_execute_on_background_thread(),
!executed_on_background_thread_);
return executed_on_background_thread_;
}
State state() const { return state_; } State state() const { return state_; }
CompilationInfo* info() const { return info_; } CompilationInfo* info() const { return info_; }
Isolate* isolate() const; Isolate* isolate() const;
...@@ -218,6 +224,7 @@ class CompilationJob { ...@@ -218,6 +224,7 @@ class CompilationJob {
const char* compiler_name_; const char* compiler_name_;
State state_; State state_;
uintptr_t stack_limit_; uintptr_t stack_limit_;
bool executed_on_background_thread_;
MUST_USE_RESULT Status UpdateState(Status status, State next_state) { MUST_USE_RESULT Status UpdateState(Status status, State next_state) {
if (status == SUCCEEDED) { if (status == SUCCEEDED) {
......
...@@ -691,7 +691,12 @@ class RuntimeCallTimer final { ...@@ -691,7 +691,12 @@ class RuntimeCallTimer final {
V(CompileDeserialize) \ V(CompileDeserialize) \
V(CompileEval) \ V(CompileEval) \
V(CompileFullCode) \ V(CompileFullCode) \
V(CompileGetFromOptimizedCodeMap) \
V(CompileGetUnoptimizedCode) \
V(CompileAnalyse) \
V(CompileIgnition) \ V(CompileIgnition) \
V(CompileBackgroundIgnition) \
V(CompileIgnitionFinalization) \
V(CompilerDispatcher) \ V(CompilerDispatcher) \
V(CompileSerialize) \ V(CompileSerialize) \
V(DeoptimizeCode) \ V(DeoptimizeCode) \
......
...@@ -41,9 +41,41 @@ class InterpreterCompilationJob final : public CompilationJob { ...@@ -41,9 +41,41 @@ class InterpreterCompilationJob final : public CompilationJob {
Status FinalizeJobImpl() final; Status FinalizeJobImpl() final;
private: private:
class TimerScope final {
public:
TimerScope(RuntimeCallStats* stats, RuntimeCallStats::CounterId counter_id)
: stats_(stats) {
if (V8_UNLIKELY(FLAG_runtime_stats)) {
RuntimeCallStats::Enter(stats_, &timer_, counter_id);
}
}
explicit TimerScope(RuntimeCallCounter* counter) : stats_(nullptr) {
if (V8_UNLIKELY(FLAG_runtime_stats)) {
timer_.Start(counter, nullptr);
}
}
~TimerScope() {
if (V8_UNLIKELY(FLAG_runtime_stats)) {
if (stats_) {
RuntimeCallStats::Leave(stats_, &timer_);
} else {
timer_.Stop();
}
}
}
private:
RuntimeCallStats* stats_;
RuntimeCallTimer timer_;
};
BytecodeGenerator* generator() { return &generator_; } BytecodeGenerator* generator() { return &generator_; }
BytecodeGenerator generator_; BytecodeGenerator generator_;
RuntimeCallStats* runtime_call_stats_;
RuntimeCallCounter background_execute_counter_;
DISALLOW_COPY_AND_ASSIGN(InterpreterCompilationJob); DISALLOW_COPY_AND_ASSIGN(InterpreterCompilationJob);
}; };
...@@ -162,7 +194,9 @@ int Interpreter::InterruptBudget() { ...@@ -162,7 +194,9 @@ int Interpreter::InterruptBudget() {
InterpreterCompilationJob::InterpreterCompilationJob(CompilationInfo* info, InterpreterCompilationJob::InterpreterCompilationJob(CompilationInfo* info,
LazyCompilationMode mode) LazyCompilationMode mode)
: CompilationJob(info->isolate(), info, "Ignition"), : CompilationJob(info->isolate(), info, "Ignition"),
generator_(info, mode) {} generator_(info, mode),
runtime_call_stats_(info->isolate()->counters()->runtime_call_stats()),
background_execute_counter_("CompileBackgroundIgnition") {}
InterpreterCompilationJob::Status InterpreterCompilationJob::PrepareJobImpl() { InterpreterCompilationJob::Status InterpreterCompilationJob::PrepareJobImpl() {
CodeGenerator::MakeCodePrologue(info(), "interpreter"); CodeGenerator::MakeCodePrologue(info(), "interpreter");
...@@ -178,11 +212,11 @@ InterpreterCompilationJob::Status InterpreterCompilationJob::PrepareJobImpl() { ...@@ -178,11 +212,11 @@ InterpreterCompilationJob::Status InterpreterCompilationJob::PrepareJobImpl() {
} }
InterpreterCompilationJob::Status InterpreterCompilationJob::ExecuteJobImpl() { InterpreterCompilationJob::Status InterpreterCompilationJob::ExecuteJobImpl() {
// TODO(5203): These timers aren't thread safe, move to using the CompilerJob TimerScope runtimeTimer =
// timers. executed_on_background_thread()
RuntimeCallTimerScope runtimeTimer(info()->isolate(), ? TimerScope(&background_execute_counter_)
&RuntimeCallStats::CompileIgnition); : TimerScope(runtime_call_stats_, &RuntimeCallStats::CompileIgnition);
TimerEventScope<TimerEventCompileIgnition> timer(info()->isolate()); // TODO(lpy): add support for background compilation RCS trace.
TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("v8.compile"), "V8.CompileIgnition"); TRACE_EVENT0(TRACE_DISABLED_BY_DEFAULT("v8.compile"), "V8.CompileIgnition");
generator()->GenerateBytecode(stack_limit()); generator()->GenerateBytecode(stack_limit());
...@@ -194,6 +228,15 @@ InterpreterCompilationJob::Status InterpreterCompilationJob::ExecuteJobImpl() { ...@@ -194,6 +228,15 @@ InterpreterCompilationJob::Status InterpreterCompilationJob::ExecuteJobImpl() {
} }
InterpreterCompilationJob::Status InterpreterCompilationJob::FinalizeJobImpl() { InterpreterCompilationJob::Status InterpreterCompilationJob::FinalizeJobImpl() {
// Add background runtime call stats.
if (V8_UNLIKELY(FLAG_runtime_stats && executed_on_background_thread())) {
runtime_call_stats_->CompileBackgroundIgnition.Add(
&background_execute_counter_);
}
RuntimeCallTimerScope runtimeTimer(
runtime_call_stats_, &RuntimeCallStats::CompileIgnitionFinalization);
Handle<BytecodeArray> bytecodes = generator()->FinalizeBytecode(isolate()); Handle<BytecodeArray> bytecodes = generator()->FinalizeBytecode(isolate());
if (generator()->HasStackOverflow()) { if (generator()->HasStackOverflow()) {
return FAILED; return FAILED;
......
...@@ -1443,6 +1443,7 @@ code is governed by a BSD-style license that can be found in the LICENSE file. ...@@ -1443,6 +1443,7 @@ code is governed by a BSD-style license that can be found in the LICENSE file.
this.total, this.total,
Group.groups.get('ic').entry(), Group.groups.get('ic').entry(),
Group.groups.get('optimize').entry(), Group.groups.get('optimize').entry(),
Group.groups.get('compile-background').entry(),
Group.groups.get('compile').entry(), Group.groups.get('compile').entry(),
Group.groups.get('parse-background').entry(), Group.groups.get('parse-background').entry(),
Group.groups.get('parse').entry(), Group.groups.get('parse').entry(),
...@@ -1651,6 +1652,8 @@ code is governed by a BSD-style license that can be found in the LICENSE file. ...@@ -1651,6 +1652,8 @@ code is governed by a BSD-style license that can be found in the LICENSE file.
Group.add('ic', new Group('IC', /.*IC_.*/, "#3366CC")); Group.add('ic', new Group('IC', /.*IC_.*/, "#3366CC"));
Group.add('optimize', new Group('Optimize', Group.add('optimize', new Group('Optimize',
/StackGuard|.*Optimize.*|.*Deoptimize.*|Recompile.*/, "#DC3912")); /StackGuard|.*Optimize.*|.*Deoptimize.*|Recompile.*/, "#DC3912"));
Group.add('compile-background', new Group('Compile-Background',
/(.*CompileBackground.*)/, "#b9a720"));
Group.add('compile', new Group('Compile', Group.add('compile', new Group('Compile',
/(^Compile.*)|(.*_Compile.*)/, "#FFAA00")); /(^Compile.*)|(.*_Compile.*)/, "#FFAA00"));
Group.add('parse-background', Group.add('parse-background',
......
...@@ -349,6 +349,7 @@ def read_stats(path, domain, args): ...@@ -349,6 +349,7 @@ def read_stats(path, domain, args):
('Group-IC', re.compile(".*IC_.*")), ('Group-IC', re.compile(".*IC_.*")),
('Group-Optimize', ('Group-Optimize',
re.compile("StackGuard|.*Optimize.*|.*Deoptimize.*|Recompile.*")), re.compile("StackGuard|.*Optimize.*|.*Deoptimize.*|Recompile.*")),
('Group-CompileBackground', re.compile("(.*CompileBackground.*)")),
('Group-Compile', re.compile("(^Compile.*)|(.*_Compile.*)")), ('Group-Compile', re.compile("(^Compile.*)|(.*_Compile.*)")),
('Group-ParseBackground', re.compile(".*ParseBackground.*")), ('Group-ParseBackground', re.compile(".*ParseBackground.*")),
('Group-Parse', re.compile(".*Parse.*")), ('Group-Parse', re.compile(".*Parse.*")),
...@@ -402,6 +403,13 @@ def read_stats(path, domain, args): ...@@ -402,6 +403,13 @@ def read_stats(path, domain, args):
group_data['time'] += entries[group_name]['time'] group_data['time'] += entries[group_name]['time']
group_data['count'] += entries[group_name]['count'] group_data['count'] += entries[group_name]['count']
entries['Group-Parse-Total'] = group_data entries['Group-Parse-Total'] = group_data
# Calculate the Compile-Total group
group_data = { 'time': 0, 'count': 0 }
for group_name, regexp in groups:
if not group_name.startswith('Group-Compile'): continue
group_data['time'] += entries[group_name]['time']
group_data['count'] += entries[group_name]['count']
entries['Group-Compile-Total'] = group_data
# Append the sums as single entries to domain. # Append the sums as single entries to domain.
for key in entries: for key in entries:
if key not in domain: domain[key] = { 'time_list': [], 'count_list': [] } if key not in domain: domain[key] = { 'time_list': [], 'count_list': [] }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment