// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. // This file contains the GPUTrace class. #ifndef GPU_COMMAND_BUFFER_SERVICE_GPU_TRACER_H_ #define GPU_COMMAND_BUFFER_SERVICE_GPU_TRACER_H_ #include #include #include #include #include "base/basictypes.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "base/threading/thread.h" #include "gpu/command_buffer/service/gles2_cmd_decoder.h" #include "gpu/gpu_export.h" namespace gfx { class GPUTimingClient; class GPUTimer; } namespace gpu { namespace gles2 { class Outputter; class GPUTrace; // Id used to keep trace namespaces separate enum GpuTracerSource { kTraceGroupInvalid = -1, kTraceCHROMIUM, kTraceDecoder, kTraceDisjoint, // Used internally. NUM_TRACER_SOURCES }; // Marker structure for a Trace. struct TraceMarker { TraceMarker(const std::string& category, const std::string& name); ~TraceMarker(); std::string category_; std::string name_; scoped_refptr trace_; }; // Traces GPU Commands. class GPU_EXPORT GPUTracer : public base::SupportsWeakPtr { public: explicit GPUTracer(gles2::GLES2Decoder* decoder); virtual ~GPUTracer(); void Destroy(bool have_context); // Scheduled processing in decoder begins. bool BeginDecoding(); // Scheduled processing in decoder ends. bool EndDecoding(); // Begin a trace marker. bool Begin(const std::string& category, const std::string& name, GpuTracerSource source); // End the last started trace marker. bool End(GpuTracerSource source); virtual bool IsTracing(); // Retrieve the name of the current open trace. // Returns empty string if no current open trace. const std::string& CurrentCategory(GpuTracerSource source) const; const std::string& CurrentName(GpuTracerSource source) const; protected: // Trace Processing. virtual scoped_refptr CreateOutputter(const std::string& name); virtual void PostTask(); void Process(); void ProcessTraces(); bool CheckDisjointStatus(); void ClearOngoingTraces(bool have_context); void IssueProcessTask(); scoped_refptr gpu_timing_client_; scoped_refptr outputter_; std::vector markers_[NUM_TRACER_SOURCES]; std::deque > finished_traces_; const unsigned char* gpu_trace_srv_category; const unsigned char* gpu_trace_dev_category; gles2::GLES2Decoder* decoder_; int64 disjoint_time_ = 0; bool gpu_executing_ = false; bool process_posted_ = false; bool began_device_traces_ = false; private: DISALLOW_COPY_AND_ASSIGN(GPUTracer); }; class Outputter : public base::RefCounted { public: virtual void TraceDevice(GpuTracerSource source, const std::string& category, const std::string& name, int64 start_time, int64 end_time) = 0; virtual void TraceServiceBegin(GpuTracerSource source, const std::string& category, const std::string& name) = 0; virtual void TraceServiceEnd(GpuTracerSource source, const std::string& category, const std::string& name) = 0; protected: virtual ~Outputter() {} friend class base::RefCounted; }; class TraceOutputter : public Outputter { public: static scoped_refptr Create(const std::string& name); void TraceDevice(GpuTracerSource source, const std::string& category, const std::string& name, int64 start_time, int64 end_time) override; void TraceServiceBegin(GpuTracerSource source, const std::string& category, const std::string& name) override; void TraceServiceEnd(GpuTracerSource source, const std::string& category, const std::string& name) override; protected: friend class base::RefCounted; explicit TraceOutputter(const std::string& name); ~TraceOutputter() override; base::Thread named_thread_; uint64 local_trace_device_id_ = 0; uint64 local_trace_service_id_ = 0; std::stack trace_service_id_stack_[NUM_TRACER_SOURCES]; private: DISALLOW_COPY_AND_ASSIGN(TraceOutputter); }; class GPU_EXPORT GPUTrace : public base::RefCounted { public: GPUTrace(scoped_refptr outputter, gfx::GPUTimingClient* gpu_timing_client, const GpuTracerSource source, const std::string& category, const std::string& name, const bool tracing_service, const bool tracing_device); void Destroy(bool have_context); void Start(); void End(); bool IsAvailable(); bool IsServiceTraceEnabled() const { return service_enabled_; } bool IsDeviceTraceEnabled() const { return device_enabled_; } void Process(); private: ~GPUTrace(); void Output(); friend class base::RefCounted; const GpuTracerSource source_ = kTraceGroupInvalid; const std::string category_; const std::string name_; scoped_refptr outputter_; scoped_ptr gpu_timer_; const bool service_enabled_ = false; const bool device_enabled_ = false; DISALLOW_COPY_AND_ASSIGN(GPUTrace); }; class ScopedGPUTrace { public: ScopedGPUTrace(GPUTracer* gpu_tracer, GpuTracerSource source, const std::string& category, const std::string& name) : gpu_tracer_(gpu_tracer), source_(source) { gpu_tracer_->Begin(category, name, source_); } ~ScopedGPUTrace() { gpu_tracer_->End(source_); } private: GPUTracer* gpu_tracer_; GpuTracerSource source_; }; } // namespace gles2 } // namespace gpu #endif // GPU_COMMAND_BUFFER_SERVICE_GPU_TRACER_H_