| OLD | NEW |
| 1 // Copyright 2013 the V8 project authors. All rights reserved. | 1 // Copyright 2013 the V8 project authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #ifndef V8_PROFILER_TICK_SAMPLE_H_ | 5 #include "src/profiler/simulator-helper.h" |
| 6 #define V8_PROFILER_TICK_SAMPLE_H_ | |
| 7 | 6 |
| 8 #include "include/v8.h" | 7 #include "src/simulator.h" |
| 9 | |
| 10 #include "src/base/platform/time.h" | |
| 11 #include "src/frames.h" | |
| 12 #include "src/globals.h" | |
| 13 | 8 |
| 14 namespace v8 { | 9 namespace v8 { |
| 15 namespace internal { | 10 namespace internal { |
| 16 | 11 |
| 17 class Isolate; | |
| 18 | |
| 19 // ---------------------------------------------------------------------------- | |
| 20 // Sampler | |
| 21 // | |
| 22 // A sampler periodically samples the state of the VM and optionally | |
| 23 // (if used for profiling) the program counter and stack pointer for | |
| 24 // the thread that created it. | |
| 25 | |
| 26 // TickSample captures the information collected for each sample. | |
| 27 struct TickSample { | |
| 28 // Internal profiling (with --prof + tools/$OS-tick-processor) wants to | |
| 29 // include the runtime function we're calling. Externally exposed tick | |
| 30 // samples don't care. | |
| 31 enum RecordCEntryFrame { kIncludeCEntryFrame, kSkipCEntryFrame }; | |
| 32 | |
| 33 TickSample() | |
| 34 : state(OTHER), | |
| 35 pc(NULL), | |
| 36 external_callback_entry(NULL), | |
| 37 frames_count(0), | |
| 38 has_external_callback(false), | |
| 39 update_stats(true) {} | |
| 40 void Init(Isolate* isolate, const v8::RegisterState& state, | |
| 41 RecordCEntryFrame record_c_entry_frame, bool update_stats); | |
| 42 static bool GetStackSample(Isolate* isolate, const v8::RegisterState& state, | |
| 43 RecordCEntryFrame record_c_entry_frame, | |
| 44 void** frames, size_t frames_limit, | |
| 45 v8::SampleInfo* sample_info); | |
| 46 StateTag state; // The state of the VM. | |
| 47 Address pc; // Instruction pointer. | |
| 48 union { | |
| 49 Address tos; // Top stack value (*sp). | |
| 50 Address external_callback_entry; | |
| 51 }; | |
| 52 static const unsigned kMaxFramesCountLog2 = 8; | |
| 53 static const unsigned kMaxFramesCount = (1 << kMaxFramesCountLog2) - 1; | |
| 54 Address stack[kMaxFramesCount]; // Call stack. | |
| 55 base::TimeTicks timestamp; | |
| 56 unsigned frames_count : kMaxFramesCountLog2; // Number of captured frames. | |
| 57 bool has_external_callback : 1; | |
| 58 bool update_stats : 1; // Whether the sample should update aggregated stats. | |
| 59 }; | |
| 60 | |
| 61 | |
| 62 #if defined(USE_SIMULATOR) | 12 #if defined(USE_SIMULATOR) |
| 63 class SimulatorHelper { | 13 bool SimulatorHelper::FillRegisters(Isolate* isolate, |
| 64 public: | 14 v8::RegisterState* state) { |
| 65 // Returns true if register values were successfully retrieved | 15 Simulator* simulator = isolate->thread_local_top()->simulator_; |
| 66 // from the simulator, otherwise returns false. | 16 // Check if there is active simulator. |
| 67 static bool FillRegisters(Isolate* isolate, v8::RegisterState* state); | 17 if (simulator == NULL) return false; |
| 68 }; | 18 #if V8_TARGET_ARCH_ARM |
| 19 if (!simulator->has_bad_pc()) { |
| 20 state->pc = reinterpret_cast<Address>(simulator->get_pc()); |
| 21 } |
| 22 state->sp = reinterpret_cast<Address>(simulator->get_register(Simulator::sp)); |
| 23 state->fp = |
| 24 reinterpret_cast<Address>(simulator->get_register(Simulator::r11)); |
| 25 #elif V8_TARGET_ARCH_ARM64 |
| 26 state->pc = reinterpret_cast<Address>(simulator->pc()); |
| 27 state->sp = reinterpret_cast<Address>(simulator->sp()); |
| 28 state->fp = reinterpret_cast<Address>(simulator->fp()); |
| 29 #elif V8_TARGET_ARCH_MIPS || V8_TARGET_ARCH_MIPS64 |
| 30 if (!simulator->has_bad_pc()) { |
| 31 state->pc = reinterpret_cast<Address>(simulator->get_pc()); |
| 32 } |
| 33 state->sp = reinterpret_cast<Address>(simulator->get_register(Simulator::sp)); |
| 34 state->fp = reinterpret_cast<Address>(simulator->get_register(Simulator::fp)); |
| 35 #elif V8_TARGET_ARCH_PPC |
| 36 if (!simulator->has_bad_pc()) { |
| 37 state->pc = reinterpret_cast<Address>(simulator->get_pc()); |
| 38 } |
| 39 state->sp = reinterpret_cast<Address>(simulator->get_register(Simulator::sp)); |
| 40 state->fp = reinterpret_cast<Address>(simulator->get_register(Simulator::fp)); |
| 41 #elif V8_TARGET_ARCH_S390 |
| 42 if (!simulator->has_bad_pc()) { |
| 43 state->pc = reinterpret_cast<Address>(simulator->get_pc()); |
| 44 } |
| 45 state->sp = reinterpret_cast<Address>(simulator->get_register(Simulator::sp)); |
| 46 state->fp = reinterpret_cast<Address>(simulator->get_register(Simulator::fp)); |
| 47 #endif |
| 48 if (state->sp == 0 || state->fp == 0) { |
| 49 // It possible that the simulator is interrupted while it is updating |
| 50 // the sp or fp register. ARM64 simulator does this in two steps: |
| 51 // first setting it to zero and then setting it to the new value. |
| 52 // Bailout if sp/fp doesn't contain the new value. |
| 53 // |
| 54 // FIXME: The above doesn't really solve the issue. |
| 55 // If a 64-bit target is executed on a 32-bit host even the final |
| 56 // write is non-atomic, so it might obtain a half of the result. |
| 57 // Moreover as long as the register set code uses memcpy (as of now), |
| 58 // it is not guaranteed to be atomic even when both host and target |
| 59 // are of same bitness. |
| 60 return false; |
| 61 } |
| 62 return true; |
| 63 } |
| 69 #endif // USE_SIMULATOR | 64 #endif // USE_SIMULATOR |
| 70 | 65 |
| 71 } // namespace internal | 66 } // namespace internal |
| 72 } // namespace v8 | 67 } // namespace v8 |
| 73 | |
| 74 #endif // V8_PROFILER_TICK_SAMPLE_H_ | |
| OLD | NEW |