• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (c) 2022 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  *     http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 
16 #include "agent/tracing_impl.h"
17 
18 #include "tooling/base/pt_events.h"
19 #include "protocol_channel.h"
20 
21 #include "ecmascript/napi/include/dfx_jsnapi.h"
22 
23 namespace panda::ecmascript::tooling {
Dispatch(const DispatchRequest & request)24 void TracingImpl::DispatcherImpl::Dispatch(const DispatchRequest &request)
25 {
26     Method method = GetMethodEnum(request.GetMethod());
27     LOG_DEBUGGER(DEBUG) << "dispatch [" << request.GetMethod() << "] to TracingImpl";
28     switch (method) {
29         case Method::END:
30             End(request);
31             break;
32         case Method::GET_CATEGORIES:
33             GetCategories(request);
34             break;
35         case Method::RECORD_CLOCK_SYNC_MARKER:
36             RecordClockSyncMarker(request);
37             break;
38         case Method::REQUEST_MEMORY_DUMP:
39             RequestMemoryDump(request);
40             break;
41         case Method::START:
42             Start(request);
43             break;
44         default:
45             SendResponse(request, DispatchResponse::Fail("Unknown method: " + request.GetMethod()));
46             break;
47     }
48 }
49 
GetMethodEnum(const std::string & method)50 TracingImpl::DispatcherImpl::Method TracingImpl::DispatcherImpl::GetMethodEnum(const std::string& method)
51 {
52     if (method == "end") {
53         return Method::END;
54     } else if (method == "getCategories") {
55         return Method::GET_CATEGORIES;
56     } else if (method == "recordClockSyncMarker") {
57         return Method::RECORD_CLOCK_SYNC_MARKER;
58     } else if (method == "requestMemoryDump") {
59         return Method::REQUEST_MEMORY_DUMP;
60     } else if (method == "start") {
61         return Method::START;
62     } else {
63         return Method::UNKNOWN;
64     }
65 }
66 
End(const DispatchRequest & request)67 void TracingImpl::DispatcherImpl::End(const DispatchRequest &request)
68 {
69     auto traceEvents = tracing_->End();
70     if (traceEvents == nullptr) {
71         LOG_DEBUGGER(ERROR) << "Transfer DFXJSNApi::StopTracing is failure";
72         SendResponse(request, DispatchResponse::Fail("Stop is failure"));
73         return;
74     }
75     SendResponse(request, DispatchResponse::Ok());
76 
77     tracing_->frontend_.DataCollected(std::move(traceEvents));
78     tracing_->frontend_.TracingComplete();
79 }
80 
GetCategories(const DispatchRequest & request)81 void TracingImpl::DispatcherImpl::GetCategories(const DispatchRequest &request)
82 {
83     std::vector<std::string> categories;
84     DispatchResponse response = tracing_->GetCategories(categories);
85     SendResponse(request, response);
86 }
87 
RecordClockSyncMarker(const DispatchRequest & request)88 void TracingImpl::DispatcherImpl::RecordClockSyncMarker(const DispatchRequest &request)
89 {
90     std::string syncId;
91     DispatchResponse response = tracing_->RecordClockSyncMarker(syncId);
92     SendResponse(request, response);
93 }
94 
RequestMemoryDump(const DispatchRequest & request)95 void TracingImpl::DispatcherImpl::RequestMemoryDump(const DispatchRequest &request)
96 {
97     std::unique_ptr<RequestMemoryDumpParams> params =
98         RequestMemoryDumpParams::Create(request.GetParams());
99     std::string dumpGuid;
100     bool success = false;
101     DispatchResponse response = tracing_->RequestMemoryDump(std::move(params), dumpGuid, success);
102     SendResponse(request, response);
103 }
104 
Start(const DispatchRequest & request)105 void TracingImpl::DispatcherImpl::Start(const DispatchRequest &request)
106 {
107     std::unique_ptr<StartParams> params =
108         StartParams::Create(request.GetParams());
109     DispatchResponse response = tracing_->Start(std::move(params));
110     SendResponse(request, response);
111 }
112 
AllowNotify() const113 bool TracingImpl::Frontend::AllowNotify() const
114 {
115     return channel_ != nullptr;
116 }
117 
BufferUsage(double percentFull,int32_t eventCount,double value)118 void TracingImpl::Frontend::BufferUsage(double percentFull, int32_t eventCount, double value)
119 {
120     if (!AllowNotify()) {
121         return;
122     }
123 
124     tooling::BufferUsage bufferUsage;
125     bufferUsage.SetPercentFull(percentFull).SetEventCount(eventCount).SetValue(value);
126     channel_->SendNotification(bufferUsage);
127 }
128 
DataCollected(std::unique_ptr<std::vector<TraceEvent>> traceEvents)129 void TracingImpl::Frontend::DataCollected(std::unique_ptr<std::vector<TraceEvent>> traceEvents)
130 {
131     if (!AllowNotify()) {
132         return;
133     }
134 
135     tooling::DataCollected dataCollected;
136     dataCollected.SetTraceEvents(std::move(traceEvents));
137 
138     channel_->SendNotification(dataCollected);
139 }
140 
TracingComplete()141 void TracingImpl::Frontend::TracingComplete()
142 {
143     if (!AllowNotify()) {
144         return;
145     }
146 
147     tooling::TracingComplete tracingComplete;
148     channel_->SendNotification(tracingComplete);
149 }
150 
End()151 std::unique_ptr<std::vector<TraceEvent>> TracingImpl::End()
152 {
153 #if defined(ECMASCRIPT_SUPPORT_TRACING)
154     uv_timer_stop(&handle_);
155 #endif
156     auto traceEvents = panda::DFXJSNApi::StopTracing(vm_);
157     return traceEvents;
158 }
159 
GetCategories(std::vector<std::string> categories)160 DispatchResponse TracingImpl::GetCategories([[maybe_unused]] std::vector<std::string> categories)
161 {
162     return DispatchResponse::Fail("GetCategories not support now.");
163 }
164 
RecordClockSyncMarker(std::string syncId)165 DispatchResponse TracingImpl::RecordClockSyncMarker([[maybe_unused]] std::string syncId)
166 {
167     return DispatchResponse::Fail("RecordClockSyncMarker not support now.");
168 }
169 
RequestMemoryDump(std::unique_ptr<RequestMemoryDumpParams> params,std::string dumpGuid,bool success)170 DispatchResponse TracingImpl::RequestMemoryDump([[maybe_unused]] std::unique_ptr<RequestMemoryDumpParams> params,
171                                                 [[maybe_unused]] std::string dumpGuid, [[maybe_unused]] bool success)
172 {
173     return DispatchResponse::Fail("RequestMemoryDump not support now.");
174 }
175 
Start(std::unique_ptr<StartParams> params)176 DispatchResponse TracingImpl::Start(std::unique_ptr<StartParams> params)
177 {
178     std::string categories = params->GetCategories();
179     if (!panda::DFXJSNApi::StartTracing(vm_, categories)) {
180         return DispatchResponse::Fail("Start tracing failed");
181     }
182 
183 #if defined(ECMASCRIPT_SUPPORT_TRACING)
184     if (params->HasBufferUsageReportingInterval()) {
185         LOG_DEBUGGER(ERROR) << "HasBufferUsageReportingInterval " << params->GetBufferUsageReportingInterval();
186         if (uv_is_active(reinterpret_cast<uv_handle_t*>(&handle_))) {
187             LOG_DEBUGGER(ERROR) << "uv_is_active!!!";
188             return DispatchResponse::Ok();
189         }
190 
191         uv_loop_t *loop = reinterpret_cast<uv_loop_t *>(vm_->GetLoop());
192         if (loop == nullptr) {
193             return DispatchResponse::Fail("Loop is nullptr");
194         }
195         uv_timer_init(loop, &handle_);
196         handle_.data = this;
197         uv_timer_start(&handle_, TracingBufferUsageReport, 0, params->GetBufferUsageReportingInterval());
198         if (DebuggerApi::IsMainThread()) {
199             uv_async_send(&loop->wq_async);
200         } else {
201             uv_work_t *work = new uv_work_t;
202             uv_queue_work(loop, work, [](uv_work_t *) { }, [](uv_work_t *work, int32_t) { delete work; });
203         }
204     }
205 #endif
206     return DispatchResponse::Ok();
207 }
208 
209 #if defined(ECMASCRIPT_SUPPORT_TRACING)
TracingBufferUsageReport(uv_timer_t * handle)210 void TracingImpl::TracingBufferUsageReport(uv_timer_t* handle)
211 {
212     TracingImpl *tracing = static_cast<TracingImpl *>(handle->data);
213     if (tracing == nullptr) {
214         LOG_DEBUGGER(ERROR) << "tracing == nullptr";
215         return;
216     }
217 
218     double percentFull = 0.0;
219     uint32_t eventCount = 0;
220     double value = 0.0;
221     panda::DFXJSNApi::GetTracingBufferUseage(tracing->vm_, percentFull, eventCount, value);
222     tracing->frontend_.BufferUsage(percentFull, eventCount, value);
223 }
224 #endif
225 }  // namespace panda::ecmascript::tooling