| /* |
| * Copyright (C) 2011 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #ifndef ART_RUNTIME_TRACE_H_ |
| #define ART_RUNTIME_TRACE_H_ |
| |
| #include <memory> |
| #include <ostream> |
| #include <set> |
| #include <string> |
| #include <vector> |
| |
| #include "atomic.h" |
| #include "base/macros.h" |
| #include "globals.h" |
| #include "instrumentation.h" |
| #include "os.h" |
| #include "safe_map.h" |
| |
| namespace art { |
| |
| namespace mirror { |
| class ArtField; |
| class ArtMethod; |
| } // namespace mirror |
| |
| class Thread; |
| |
| enum TracingMode { |
| kTracingInactive, |
| kMethodTracingActive, |
| kSampleProfilingActive, |
| }; |
| |
| class Trace FINAL : public instrumentation::InstrumentationListener { |
| public: |
| enum TraceFlag { |
| kTraceCountAllocs = 1, |
| }; |
| |
| static void SetDefaultClockSource(TraceClockSource clock_source); |
| |
| static void Start(const char* trace_filename, int trace_fd, int buffer_size, int flags, |
| bool direct_to_ddms, bool sampling_enabled, int interval_us) |
| LOCKS_EXCLUDED(Locks::mutator_lock_, |
| Locks::thread_list_lock_, |
| Locks::thread_suspend_count_lock_, |
| Locks::trace_lock_); |
| static void Stop() |
| LOCKS_EXCLUDED(Locks::mutator_lock_, |
| Locks::thread_list_lock_, |
| Locks::trace_lock_); |
| static void Shutdown() LOCKS_EXCLUDED(Locks::trace_lock_); |
| static TracingMode GetMethodTracingMode() LOCKS_EXCLUDED(Locks::trace_lock_); |
| |
| bool UseWallClock(); |
| bool UseThreadCpuClock(); |
| void MeasureClockOverhead(); |
| uint32_t GetClockOverheadNanoSeconds(); |
| |
| void CompareAndUpdateStackTrace(Thread* thread, std::vector<mirror::ArtMethod*>* stack_trace) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_); |
| |
| // InstrumentationListener implementation. |
| void MethodEntered(Thread* thread, mirror::Object* this_object, |
| mirror::ArtMethod* method, uint32_t dex_pc) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) OVERRIDE; |
| void MethodExited(Thread* thread, mirror::Object* this_object, |
| mirror::ArtMethod* method, uint32_t dex_pc, |
| const JValue& return_value) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) OVERRIDE; |
| void MethodUnwind(Thread* thread, mirror::Object* this_object, |
| mirror::ArtMethod* method, uint32_t dex_pc) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) OVERRIDE; |
| void DexPcMoved(Thread* thread, mirror::Object* this_object, |
| mirror::ArtMethod* method, uint32_t new_dex_pc) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) OVERRIDE; |
| void FieldRead(Thread* thread, mirror::Object* this_object, |
| mirror::ArtMethod* method, uint32_t dex_pc, mirror::ArtField* field) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) OVERRIDE; |
| void FieldWritten(Thread* thread, mirror::Object* this_object, |
| mirror::ArtMethod* method, uint32_t dex_pc, mirror::ArtField* field, |
| const JValue& field_value) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) OVERRIDE; |
| void ExceptionCaught(Thread* thread, mirror::Throwable* exception_object) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) OVERRIDE; |
| void BackwardBranch(Thread* thread, mirror::ArtMethod* method, int32_t dex_pc_offset) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) OVERRIDE; |
| // Reuse an old stack trace if it exists, otherwise allocate a new one. |
| static std::vector<mirror::ArtMethod*>* AllocStackTrace(); |
| // Clear and store an old stack trace for later use. |
| static void FreeStackTrace(std::vector<mirror::ArtMethod*>* stack_trace); |
| // Save id and name of a thread before it exits. |
| static void StoreExitingThreadInfo(Thread* thread); |
| |
| private: |
| explicit Trace(File* trace_file, int buffer_size, int flags, bool sampling_enabled); |
| |
| // The sampling interval in microseconds is passed as an argument. |
| static void* RunSamplingThread(void* arg) LOCKS_EXCLUDED(Locks::trace_lock_); |
| |
| void FinishTracing() SHARED_LOCKS_REQUIRED(Locks::mutator_lock_); |
| |
| void ReadClocks(Thread* thread, uint32_t* thread_clock_diff, uint32_t* wall_clock_diff); |
| |
| void LogMethodTraceEvent(Thread* thread, mirror::ArtMethod* method, |
| instrumentation::Instrumentation::InstrumentationEvent event, |
| uint32_t thread_clock_diff, uint32_t wall_clock_diff); |
| |
| // Methods to output traced methods and threads. |
| void GetVisitedMethods(size_t end_offset, std::set<mirror::ArtMethod*>* visited_methods); |
| void DumpMethodList(std::ostream& os, const std::set<mirror::ArtMethod*>& visited_methods) |
| SHARED_LOCKS_REQUIRED(Locks::mutator_lock_); |
| void DumpThreadList(std::ostream& os) LOCKS_EXCLUDED(Locks::thread_list_lock_); |
| |
| // Singleton instance of the Trace or NULL when no method tracing is active. |
| static Trace* volatile the_trace_ GUARDED_BY(Locks::trace_lock_); |
| |
| // The default profiler clock source. |
| static TraceClockSource default_clock_source_; |
| |
| // Sampling thread, non-zero when sampling. |
| static pthread_t sampling_pthread_; |
| |
| // Used to remember an unused stack trace to avoid re-allocation during sampling. |
| static std::unique_ptr<std::vector<mirror::ArtMethod*>> temp_stack_trace_; |
| |
| // File to write trace data out to, NULL if direct to ddms. |
| std::unique_ptr<File> trace_file_; |
| |
| // Buffer to store trace data. |
| std::unique_ptr<uint8_t> buf_; |
| |
| // Flags enabling extra tracing of things such as alloc counts. |
| const int flags_; |
| |
| // True if traceview should sample instead of instrumenting method entry/exit. |
| const bool sampling_enabled_; |
| |
| const TraceClockSource clock_source_; |
| |
| // Size of buf_. |
| const int buffer_size_; |
| |
| // Time trace was created. |
| const uint64_t start_time_; |
| |
| // Clock overhead. |
| const uint32_t clock_overhead_ns_; |
| |
| // Offset into buf_. |
| AtomicInteger cur_offset_; |
| |
| // Did we overflow the buffer recording traces? |
| bool overflow_; |
| |
| // Map of thread ids and names that have already exited. |
| SafeMap<pid_t, std::string> exited_threads_; |
| |
| DISALLOW_COPY_AND_ASSIGN(Trace); |
| }; |
| |
| } // namespace art |
| |
| #endif // ART_RUNTIME_TRACE_H_ |