Commit 9b157602 authored by Alexei Filippov's avatar Alexei Filippov Committed by Commit Bot

[runtime-call-stats] Fix a long standing crash in RuntimeCallStats::Leave

There must be a matching Leave for each Enter. Otherwise it ends up
with a dead stack-allocated object in the timer chain.

Drive-by: There was also a bug in
RuntimeCallTimerScope::RuntimeCallTimerScope(HeapObject* ...) did create a
local object instead of calling an overloaded constructor.

BUG=chromium:669329

Change-Id: I9aa1c574a854af8beab3d8097efab3a726ad1c8d
Reviewed-on: https://chromium-review.googlesource.com/634511
Commit-Queue: Alexei Filippov <alph@chromium.org>
Reviewed-by: 's avatarCamillo Bruni <cbruni@chromium.org>
Reviewed-by: 's avatarRoss McIlroy <rmcilroy@chromium.org>
Cr-Commit-Position: refs/heads/master@{#47613}
parent e5df5bd0
...@@ -65,20 +65,25 @@ base::TimeTicks RuntimeCallTimer::Now() { ...@@ -65,20 +65,25 @@ base::TimeTicks RuntimeCallTimer::Now() {
RuntimeCallTimerScope::RuntimeCallTimerScope( RuntimeCallTimerScope::RuntimeCallTimerScope(
Isolate* isolate, RuntimeCallStats::CounterId counter_id) { Isolate* isolate, RuntimeCallStats::CounterId counter_id) {
if (V8_UNLIKELY(FLAG_runtime_stats)) { if (V8_LIKELY(!FLAG_runtime_stats)) return;
Initialize(isolate->counters()->runtime_call_stats(), counter_id); stats_ = isolate->counters()->runtime_call_stats();
} RuntimeCallStats::Enter(stats_, &timer_, counter_id);
} }
RuntimeCallTimerScope::RuntimeCallTimerScope( RuntimeCallTimerScope::RuntimeCallTimerScope(
HeapObject* heap_object, RuntimeCallStats::CounterId counter_id) { HeapObject* heap_object, RuntimeCallStats::CounterId counter_id)
RuntimeCallTimerScope(heap_object->GetIsolate(), counter_id); : RuntimeCallTimerScope(heap_object->GetIsolate(), counter_id) {}
}
RuntimeCallTimerScope::RuntimeCallTimerScope( RuntimeCallTimerScope::RuntimeCallTimerScope(
RuntimeCallStats* stats, RuntimeCallStats::CounterId counter_id) { RuntimeCallStats* stats, RuntimeCallStats::CounterId counter_id) {
if (V8_UNLIKELY(FLAG_runtime_stats)) { if (V8_LIKELY(!FLAG_runtime_stats)) return;
Initialize(stats, counter_id); stats_ = stats;
RuntimeCallStats::Enter(stats_, &timer_, counter_id);
}
RuntimeCallTimerScope::~RuntimeCallTimerScope() {
if (V8_UNLIKELY(stats_ != nullptr)) {
RuntimeCallStats::Leave(stats_, &timer_);
} }
} }
......
...@@ -485,26 +485,10 @@ void RuntimeCallStats::Enter(RuntimeCallStats* stats, RuntimeCallTimer* timer, ...@@ -485,26 +485,10 @@ void RuntimeCallStats::Enter(RuntimeCallStats* stats, RuntimeCallTimer* timer,
// static // static
void RuntimeCallStats::Leave(RuntimeCallStats* stats, RuntimeCallTimer* timer) { void RuntimeCallStats::Leave(RuntimeCallStats* stats, RuntimeCallTimer* timer) {
if (stats->current_timer_.Value() == timer) { CHECK(stats->current_timer_.Value() == timer);
stats->current_timer_.SetValue(timer->Stop()); stats->current_timer_.SetValue(timer->Stop());
} else {
// Must be a Threading cctest. Walk the chain of Timers to find the
// buried one that's leaving. We don't care about keeping nested timings
// accurate, just avoid crashing by keeping the chain intact.
RuntimeCallTimer* next = stats->current_timer_.Value();
while (next && next->parent() != timer) next = next->parent();
if (next == nullptr) return;
next->set_parent(timer->Stop());
}
{
RuntimeCallTimer* cur_timer = stats->current_timer_.Value(); RuntimeCallTimer* cur_timer = stats->current_timer_.Value();
if (cur_timer == nullptr) { stats->current_counter_.SetValue(cur_timer ? cur_timer->counter() : nullptr);
stats->current_counter_.SetValue(nullptr);
} else {
stats->current_counter_.SetValue(cur_timer->counter());
}
}
} }
void RuntimeCallStats::Add(RuntimeCallStats* other) { void RuntimeCallStats::Add(RuntimeCallStats* other) {
......
...@@ -971,20 +971,9 @@ class RuntimeCallTimerScope { ...@@ -971,20 +971,9 @@ class RuntimeCallTimerScope {
RuntimeCallStats::CounterId counter_id); RuntimeCallStats::CounterId counter_id);
inline RuntimeCallTimerScope(RuntimeCallStats* stats, inline RuntimeCallTimerScope(RuntimeCallStats* stats,
RuntimeCallStats::CounterId counter_id); RuntimeCallStats::CounterId counter_id);
inline ~RuntimeCallTimerScope();
inline ~RuntimeCallTimerScope() {
if (V8_UNLIKELY(stats_ != nullptr)) {
RuntimeCallStats::Leave(stats_, &timer_);
}
}
private: private:
V8_INLINE void Initialize(RuntimeCallStats* stats,
RuntimeCallStats::CounterId counter_id) {
stats_ = stats;
RuntimeCallStats::Enter(stats_, &timer_, counter_id);
}
RuntimeCallStats* stats_ = nullptr; RuntimeCallStats* stats_ = nullptr;
RuntimeCallTimer timer_; RuntimeCallTimer timer_;
}; };
......
...@@ -36,31 +36,32 @@ class InterpreterCompilationJob final : public CompilationJob { ...@@ -36,31 +36,32 @@ class InterpreterCompilationJob final : public CompilationJob {
class TimerScope final { class TimerScope final {
public: public:
TimerScope(RuntimeCallStats* stats, RuntimeCallStats::CounterId counter_id) TimerScope(RuntimeCallStats* stats, RuntimeCallStats::CounterId counter_id)
: stats_(stats) { : stats_(stats), runtime_stats_enabled_(FLAG_runtime_stats) {
if (V8_UNLIKELY(FLAG_runtime_stats)) { if (V8_UNLIKELY(runtime_stats_enabled_)) {
RuntimeCallStats::Enter(stats_, &timer_, counter_id); RuntimeCallStats::Enter(stats_, &timer_, counter_id);
} }
} }
explicit TimerScope(RuntimeCallCounter* counter) : stats_(nullptr) { explicit TimerScope(RuntimeCallCounter* counter)
if (V8_UNLIKELY(FLAG_runtime_stats)) { : stats_(nullptr), runtime_stats_enabled_(FLAG_runtime_stats) {
if (V8_UNLIKELY(runtime_stats_enabled_)) {
timer_.Start(counter, nullptr); timer_.Start(counter, nullptr);
} }
} }
~TimerScope() { ~TimerScope() {
if (V8_UNLIKELY(FLAG_runtime_stats)) { if (V8_LIKELY(!runtime_stats_enabled_)) return;
if (stats_) { if (stats_) {
RuntimeCallStats::Leave(stats_, &timer_); RuntimeCallStats::Leave(stats_, &timer_);
} else { } else {
timer_.Stop(); timer_.Stop();
} }
} }
}
private: private:
RuntimeCallStats* stats_; RuntimeCallStats* stats_;
RuntimeCallTimer timer_; RuntimeCallTimer timer_;
bool runtime_stats_enabled_;
}; };
BytecodeGenerator* generator() { return &generator_; } BytecodeGenerator* generator() { return &generator_; }
......
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