From c0fca4e8c819903070243490fceeacbb22782e70 Mon Sep 17 00:00:00 2001 From: "yurys@chromium.org" Date: Fri, 19 Apr 2013 11:55:01 +0000 Subject: [PATCH] Revert r14252 as it broke --prof for some cases R=jkummerow BUG=v8:2642 Review URL: https://codereview.chromium.org/14367020 git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14348 ce2b1a6d-e550-0410-aec6-3dcde31c8c00 --- src/log.cc | 8 +++++- src/profile-generator.cc | 2 +- src/sampler.cc | 11 +++++++- src/sampler.h | 9 +++++-- test/cctest/test-cpu-profiler.cc | 2 ++ test/cctest/test-log-stack-tracer.cc | 4 +-- test/cctest/test-profile-generator.cc | 3 +++ .../tools/tickprocessor-test-func-info.log | 6 ++--- test/mjsunit/tools/tickprocessor-test.log | 26 +++++++++---------- tools/tickprocessor.js | 21 ++++++++++----- 10 files changed, 63 insertions(+), 29 deletions(-) diff --git a/src/log.cc b/src/log.cc index 55f5637d55..57abdefbab 100644 --- a/src/log.cc +++ b/src/log.cc @@ -1448,7 +1448,13 @@ void Logger::TickEvent(TickSample* sample, bool overflow) { msg.Append(','); msg.AppendAddress(sample->sp); msg.Append(",%ld", static_cast(OS::Ticks() - epoch_)); - msg.AppendAddress(sample->external_callback); + if (sample->has_external_callback) { + msg.Append(",1,"); + msg.AppendAddress(sample->external_callback); + } else { + msg.Append(",0,"); + msg.AppendAddress(sample->tos); + } msg.Append(",%d", static_cast(sample->state)); if (overflow) { msg.Append(",overflow"); diff --git a/src/profile-generator.cc b/src/profile-generator.cc index eacabeff4f..e4c750cb83 100644 --- a/src/profile-generator.cc +++ b/src/profile-generator.cc @@ -894,7 +894,7 @@ void ProfileGenerator::RecordTickSample(const TickSample& sample) { if (sample.pc != NULL) { *entry++ = code_map_.FindEntry(sample.pc); - if (sample.external_callback) { + if (sample.has_external_callback) { // Don't use PC when in external callback code, as it can point // inside callback's code, and we will erroneously report // that a callback calls itself. diff --git a/src/sampler.cc b/src/sampler.cc index 948b054073..e271470bd2 100644 --- a/src/sampler.cc +++ b/src/sampler.cc @@ -636,7 +636,16 @@ DISABLE_ASAN void TickSample::Trace(Isolate* isolate) { return; } - external_callback = isolate->external_callback(); + const Address callback = isolate->external_callback(); + if (callback != NULL) { + external_callback = callback; + has_external_callback = true; + } else { + // Sample potential return address value for frameless invocation of + // stubs (we'll figure out later, if this value makes sense). + tos = Memory::Address_at(sp); + has_external_callback = false; + } SafeStackTraceFrameIterator it(isolate, fp, sp, sp, js_entry_sp); int i = 0; diff --git a/src/sampler.h b/src/sampler.h index a76d8b9a57..1d9ac8723b 100644 --- a/src/sampler.h +++ b/src/sampler.h @@ -51,16 +51,21 @@ struct TickSample { sp(NULL), fp(NULL), external_callback(NULL), - frames_count(0) {} + frames_count(0), + has_external_callback(false) {} void Trace(Isolate* isolate); StateTag state; // The state of the VM. Address pc; // Instruction pointer. Address sp; // Stack pointer. Address fp; // Frame pointer. - Address external_callback; + union { + Address tos; // Top stack value (*sp). + Address external_callback; + }; static const int kMaxFramesCount = 64; Address stack[kMaxFramesCount]; // Call stack. int frames_count : 8; // Number of captured frames. + bool has_external_callback : 1; }; class Sampler { diff --git a/test/cctest/test-cpu-profiler.cc b/test/cctest/test-cpu-profiler.cc index 2eece46230..40b5b79125 100644 --- a/test/cctest/test-cpu-profiler.cc +++ b/test/cctest/test-cpu-profiler.cc @@ -64,6 +64,7 @@ static void EnqueueTickSampleEvent(ProfilerEventsProcessor* proc, i::Address frame3 = NULL) { i::TickSample* sample = proc->TickSampleEvent(); sample->pc = frame1; + sample->tos = frame1; sample->frames_count = 0; if (frame2 != NULL) { sample->stack[0] = frame2; @@ -238,6 +239,7 @@ TEST(Issue1398) { i::TickSample* sample = processor.TickSampleEvent(); sample->pc = ToAddress(0x1200); + sample->tos = 0; sample->frames_count = i::TickSample::kMaxFramesCount; for (int i = 0; i < sample->frames_count; ++i) { sample->stack[i] = ToAddress(0x1200); diff --git a/test/cctest/test-log-stack-tracer.cc b/test/cctest/test-log-stack-tracer.cc index 5bfc1d36be..ca6c7aea47 100644 --- a/test/cctest/test-log-stack-tracer.cc +++ b/test/cctest/test-log-stack-tracer.cc @@ -286,7 +286,7 @@ TEST(CFromJSStackTrace) { // DoTrace(EBP) [native] // TickSample::Trace - CHECK(sample.external_callback); + CHECK(sample.has_external_callback); CHECK_EQ(FUNCTION_ADDR(TraceExtension::Trace), sample.external_callback); // Stack tracing will start from the first JS function, i.e. "JSFuncDoTrace" @@ -336,7 +336,7 @@ TEST(PureJSStackTrace) { // TickSample::Trace // - CHECK(sample.external_callback); + CHECK(sample.has_external_callback); CHECK_EQ(FUNCTION_ADDR(TraceExtension::JSTrace), sample.external_callback); // Stack sampling will start from the caller of JSFuncDoTrace, i.e. "JSTrace" diff --git a/test/cctest/test-profile-generator.cc b/test/cctest/test-profile-generator.cc index 0682fbcdb1..56b1788a85 100644 --- a/test/cctest/test-profile-generator.cc +++ b/test/cctest/test-profile-generator.cc @@ -624,11 +624,13 @@ TEST(RecordTickSample) { // -> ccc -> aaa - sample3 TickSample sample1; sample1.pc = ToAddress(0x1600); + sample1.tos = ToAddress(0x1500); sample1.stack[0] = ToAddress(0x1510); sample1.frames_count = 1; generator.RecordTickSample(sample1); TickSample sample2; sample2.pc = ToAddress(0x1925); + sample2.tos = ToAddress(0x1900); sample2.stack[0] = ToAddress(0x1780); sample2.stack[1] = ToAddress(0x10000); // non-existent. sample2.stack[2] = ToAddress(0x1620); @@ -636,6 +638,7 @@ TEST(RecordTickSample) { generator.RecordTickSample(sample2); TickSample sample3; sample3.pc = ToAddress(0x1510); + sample3.tos = ToAddress(0x1500); sample3.stack[0] = ToAddress(0x1910); sample3.stack[1] = ToAddress(0x1610); sample3.frames_count = 2; diff --git a/test/mjsunit/tools/tickprocessor-test-func-info.log b/test/mjsunit/tools/tickprocessor-test-func-info.log index fccd87e744..5e64dc09e3 100644 --- a/test/mjsunit/tools/tickprocessor-test-func-info.log +++ b/test/mjsunit/tools/tickprocessor-test-func-info.log @@ -5,7 +5,7 @@ profiler,"begin",1 code-creation,Stub,0,0x424260,348,"CompareStub_GE" code-creation,LazyCompile,0,0x2a8100,18535,"DrawQube 3d-cube.js:188",0xf43abcac, code-creation,LazyCompile,0,0x480100,3908,"DrawLine 3d-cube.js:17",0xf43abc50, -tick,0x424284,0xbfffeea0,0,0,0,0x480600,0x2aaaa5 -tick,0x42429f,0xbfffed88,0,0,0,0x480600,0x2aacb4 -tick,0x48063d,0xbfffec7c,0,0,0,0x2d0f7c,0x2aaec6 +tick,0x424284,0xbfffeea0,0,0,0x480600,0,0x2aaaa5 +tick,0x42429f,0xbfffed88,0,0,0x480600,0,0x2aacb4 +tick,0x48063d,0xbfffec7c,0,0,0x2d0f7c,0,0x2aaec6 profiler,"end" diff --git a/test/mjsunit/tools/tickprocessor-test.log b/test/mjsunit/tools/tickprocessor-test.log index 2733513846..5ddad89a55 100644 --- a/test/mjsunit/tools/tickprocessor-test.log +++ b/test/mjsunit/tools/tickprocessor-test.log @@ -9,17 +9,17 @@ code-creation,LazyCompile,0,0xf541d120,145,"exp native math.js:41" function-creation,0xf441d280,0xf541d120 code-creation,LoadIC,0,0xf541d280,117,"j" code-creation,LoadIC,0,0xf541d360,63,"i" -tick,0x80f82d1,0xffdfe880,0,0,0,0xf541ce5c -tick,0x80f89a1,0xffdfecf0,0,0,0,0xf541ce5c -tick,0x8123b5c,0xffdff1a0,0,0,0,0xf541d1a1,0xf541ceea -tick,0x8123b65,0xffdff1a0,0,0,0,0xf541d1a1,0xf541ceea -tick,0xf541d2be,0xffdff1e4,0,0,0 -tick,0xf541d320,0xffdff1dc,0,0,0 -tick,0xf541d384,0xffdff1d8,0,0,0 -tick,0xf7db94da,0xffdff0ec,0,0,0,0xf541d1a1,0xf541ceea -tick,0xf7db951c,0xffdff0f0,0,0,0,0xf541d1a1,0xf541ceea -tick,0xf7dbc508,0xffdff14c,0,0,0,0xf541d1a1,0xf541ceea -tick,0xf7dbff21,0xffdff198,0,0,0,0xf541d1a1,0xf541ceea -tick,0xf7edec90,0xffdff0ec,0,0,0,0xf541d1a1,0xf541ceea -tick,0xffffe402,0xffdff488,0,0,0 +tick,0x80f82d1,0xffdfe880,0,0,0,0,0xf541ce5c +tick,0x80f89a1,0xffdfecf0,0,0,0,0,0xf541ce5c +tick,0x8123b5c,0xffdff1a0,0,0,0,0,0xf541d1a1,0xf541ceea +tick,0x8123b65,0xffdff1a0,0,0,0,0,0xf541d1a1,0xf541ceea +tick,0xf541d2be,0xffdff1e4,0,0,0,0 +tick,0xf541d320,0xffdff1dc,0,0,0,0 +tick,0xf541d384,0xffdff1d8,0,0,0,0 +tick,0xf7db94da,0xffdff0ec,0,0,0,0,0xf541d1a1,0xf541ceea +tick,0xf7db951c,0xffdff0f0,0,0,0,0,0xf541d1a1,0xf541ceea +tick,0xf7dbc508,0xffdff14c,0,0,0,0,0xf541d1a1,0xf541ceea +tick,0xf7dbff21,0xffdff198,0,0,0,0,0xf541d1a1,0xf541ceea +tick,0xf7edec90,0xffdff0ec,0,0,0,0,0xf541d1a1,0xf541ceea +tick,0xffffe402,0xffdff488,0,0,0,0 profiler,"end" diff --git a/tools/tickprocessor.js b/tools/tickprocessor.js index 0ffe7342a9..c9ee1011f0 100644 --- a/tools/tickprocessor.js +++ b/tools/tickprocessor.js @@ -170,7 +170,7 @@ function TickProcessor( processor: this.processSnapshotPosition }, 'tick': { parsers: [parseInt, parseInt, parseInt, parseInt, - parseInt, 'var-args'], + parseInt, parseInt, 'var-args'], processor: this.processTick }, 'heap-sample-begin': { parsers: [null, null, parseInt], processor: this.processHeapSampleBegin }, @@ -368,7 +368,8 @@ TickProcessor.prototype.includeTick = function(vmState) { TickProcessor.prototype.processTick = function(pc, sp, ns_since_start, - external_callback, + is_external_callback, + tos_or_external_callback, vmState, stack) { this.distortion += this.distortion_per_entry; @@ -382,15 +383,23 @@ TickProcessor.prototype.processTick = function(pc, this.ticks_.excluded++; return; } - if (external_callback) { + if (is_external_callback) { // Don't use PC when in external callback code, as it can point // inside callback's code, and we will erroneously report // that a callback calls itself. Instead we use tos_or_external_callback, // as simply resetting PC will produce unaccounted ticks. - pc = 0; - } + pc = tos_or_external_callback; + tos_or_external_callback = 0; + } else if (tos_or_external_callback) { + // Find out, if top of stack was pointing inside a JS function + // meaning that we have encountered a frameless invocation. + var funcEntry = this.profile_.findEntry(tos_or_external_callback); + if (!funcEntry || !funcEntry.isJSFunction || !funcEntry.isJSFunction()) { + tos_or_external_callback = 0; + } + } - this.profile_.recordTick(this.processStack(pc, external_callback, stack)); + this.profile_.recordTick(this.processStack(pc, tos_or_external_callback, stack)); };