1*c8dee2aaSAndroid Build Coastguard Worker /*
2*c8dee2aaSAndroid Build Coastguard Worker * Copyright 2014 Google Inc.
3*c8dee2aaSAndroid Build Coastguard Worker *
4*c8dee2aaSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license that can be
5*c8dee2aaSAndroid Build Coastguard Worker * found in the LICENSE file.
6*c8dee2aaSAndroid Build Coastguard Worker */
7*c8dee2aaSAndroid Build Coastguard Worker
8*c8dee2aaSAndroid Build Coastguard Worker #include "include/utils/SkEventTracer.h"
9*c8dee2aaSAndroid Build Coastguard Worker
10*c8dee2aaSAndroid Build Coastguard Worker #include "include/private/base/SkOnce.h"
11*c8dee2aaSAndroid Build Coastguard Worker
12*c8dee2aaSAndroid Build Coastguard Worker #include <stdlib.h>
13*c8dee2aaSAndroid Build Coastguard Worker #include <atomic>
14*c8dee2aaSAndroid Build Coastguard Worker
15*c8dee2aaSAndroid Build Coastguard Worker class SkDefaultEventTracer : public SkEventTracer {
16*c8dee2aaSAndroid Build Coastguard Worker SkEventTracer::Handle
addTraceEvent(char phase,const uint8_t * categoryEnabledFlag,const char * name,uint64_t id,int numArgs,const char ** argNames,const uint8_t * argTypes,const uint64_t * argValues,uint8_t flags)17*c8dee2aaSAndroid Build Coastguard Worker addTraceEvent(char phase,
18*c8dee2aaSAndroid Build Coastguard Worker const uint8_t* categoryEnabledFlag,
19*c8dee2aaSAndroid Build Coastguard Worker const char* name,
20*c8dee2aaSAndroid Build Coastguard Worker uint64_t id,
21*c8dee2aaSAndroid Build Coastguard Worker int numArgs,
22*c8dee2aaSAndroid Build Coastguard Worker const char** argNames,
23*c8dee2aaSAndroid Build Coastguard Worker const uint8_t* argTypes,
24*c8dee2aaSAndroid Build Coastguard Worker const uint64_t* argValues,
25*c8dee2aaSAndroid Build Coastguard Worker uint8_t flags) override { return 0; }
26*c8dee2aaSAndroid Build Coastguard Worker
27*c8dee2aaSAndroid Build Coastguard Worker void
updateTraceEventDuration(const uint8_t * categoryEnabledFlag,const char * name,SkEventTracer::Handle handle)28*c8dee2aaSAndroid Build Coastguard Worker updateTraceEventDuration(const uint8_t* categoryEnabledFlag,
29*c8dee2aaSAndroid Build Coastguard Worker const char* name,
30*c8dee2aaSAndroid Build Coastguard Worker SkEventTracer::Handle handle) override {}
31*c8dee2aaSAndroid Build Coastguard Worker
getCategoryGroupEnabled(const char * name)32*c8dee2aaSAndroid Build Coastguard Worker const uint8_t* getCategoryGroupEnabled(const char* name) override {
33*c8dee2aaSAndroid Build Coastguard Worker static uint8_t no = 0;
34*c8dee2aaSAndroid Build Coastguard Worker return &no;
35*c8dee2aaSAndroid Build Coastguard Worker }
getCategoryGroupName(const uint8_t * categoryEnabledFlag)36*c8dee2aaSAndroid Build Coastguard Worker const char* getCategoryGroupName(
37*c8dee2aaSAndroid Build Coastguard Worker const uint8_t* categoryEnabledFlag) override {
38*c8dee2aaSAndroid Build Coastguard Worker static const char* stub = "stub";
39*c8dee2aaSAndroid Build Coastguard Worker return stub;
40*c8dee2aaSAndroid Build Coastguard Worker }
41*c8dee2aaSAndroid Build Coastguard Worker
42*c8dee2aaSAndroid Build Coastguard Worker // The default tracer does not yet support splitting up trace output into sections.
newTracingSection(const char * name)43*c8dee2aaSAndroid Build Coastguard Worker void newTracingSection(const char* name) override {}
44*c8dee2aaSAndroid Build Coastguard Worker };
45*c8dee2aaSAndroid Build Coastguard Worker
46*c8dee2aaSAndroid Build Coastguard Worker // We prefer gUserTracer if it's been set, otherwise we fall back on a default tracer;
47*c8dee2aaSAndroid Build Coastguard Worker static std::atomic<SkEventTracer*> gUserTracer{nullptr};
48*c8dee2aaSAndroid Build Coastguard Worker
SetInstance(SkEventTracer * tracer,bool leakTracer)49*c8dee2aaSAndroid Build Coastguard Worker bool SkEventTracer::SetInstance(SkEventTracer* tracer, bool leakTracer) {
50*c8dee2aaSAndroid Build Coastguard Worker SkEventTracer* expected = nullptr;
51*c8dee2aaSAndroid Build Coastguard Worker if (!gUserTracer.compare_exchange_strong(expected, tracer)) {
52*c8dee2aaSAndroid Build Coastguard Worker delete tracer;
53*c8dee2aaSAndroid Build Coastguard Worker return false;
54*c8dee2aaSAndroid Build Coastguard Worker }
55*c8dee2aaSAndroid Build Coastguard Worker // If leaking the tracer is accepted then there is no need to install
56*c8dee2aaSAndroid Build Coastguard Worker // the atexit.
57*c8dee2aaSAndroid Build Coastguard Worker if (!leakTracer) {
58*c8dee2aaSAndroid Build Coastguard Worker atexit([]() { delete gUserTracer.load(); });
59*c8dee2aaSAndroid Build Coastguard Worker }
60*c8dee2aaSAndroid Build Coastguard Worker return true;
61*c8dee2aaSAndroid Build Coastguard Worker }
62*c8dee2aaSAndroid Build Coastguard Worker
GetInstance()63*c8dee2aaSAndroid Build Coastguard Worker SkEventTracer* SkEventTracer::GetInstance() {
64*c8dee2aaSAndroid Build Coastguard Worker if (auto tracer = gUserTracer.load(std::memory_order_acquire)) {
65*c8dee2aaSAndroid Build Coastguard Worker return tracer;
66*c8dee2aaSAndroid Build Coastguard Worker }
67*c8dee2aaSAndroid Build Coastguard Worker static SkOnce once;
68*c8dee2aaSAndroid Build Coastguard Worker static SkDefaultEventTracer* defaultTracer;
69*c8dee2aaSAndroid Build Coastguard Worker once([] { defaultTracer = new SkDefaultEventTracer; });
70*c8dee2aaSAndroid Build Coastguard Worker return defaultTracer;
71*c8dee2aaSAndroid Build Coastguard Worker }
72