1 /*
2 * Copyright (c) 2022-2024 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 "core/common/layout_inspector.h"
17
18 #include "include/core/SkImage.h"
19 #include "include/core/SkString.h"
20 #include "include/utils/SkBase64.h"
21
22 #include "connect_server_manager.h"
23
24 #include "adapter/ohos/osal/pixel_map_ohos.h"
25 #include "adapter/ohos/entrance/subwindow/subwindow_ohos.h"
26 #include "base/thread/background_task_executor.h"
27 #include "core/common/ace_engine.h"
28 #include "core/common/connect_server_manager.h"
29 #include "core/components_v2/inspector/inspector.h"
30 #include "base/websocket/websocket_manager.h"
31 #include "frameworks/base/log/ace_checker.h"
32
33 namespace OHOS::Ace {
34
35 namespace {
36
ColorSpaceToSkColorSpace(const RefPtr<PixelMap> & pixmap)37 sk_sp<SkColorSpace> ColorSpaceToSkColorSpace(const RefPtr<PixelMap>& pixmap)
38 {
39 return SkColorSpace::MakeSRGB();
40 }
41
AlphaTypeToSkAlphaType(const RefPtr<PixelMap> & pixmap)42 SkAlphaType AlphaTypeToSkAlphaType(const RefPtr<PixelMap>& pixmap)
43 {
44 switch (pixmap->GetAlphaType()) {
45 case AlphaType::IMAGE_ALPHA_TYPE_UNKNOWN:
46 return SkAlphaType::kUnknown_SkAlphaType;
47 case AlphaType::IMAGE_ALPHA_TYPE_OPAQUE:
48 return SkAlphaType::kOpaque_SkAlphaType;
49 case AlphaType::IMAGE_ALPHA_TYPE_PREMUL:
50 return SkAlphaType::kPremul_SkAlphaType;
51 case AlphaType::IMAGE_ALPHA_TYPE_UNPREMUL:
52 return SkAlphaType::kUnpremul_SkAlphaType;
53 default:
54 return SkAlphaType::kUnknown_SkAlphaType;
55 }
56 }
57
PixelFormatToSkColorType(const RefPtr<PixelMap> & pixmap)58 SkColorType PixelFormatToSkColorType(const RefPtr<PixelMap>& pixmap)
59 {
60 switch (pixmap->GetPixelFormat()) {
61 case PixelFormat::RGB_565:
62 return SkColorType::kRGB_565_SkColorType;
63 case PixelFormat::RGBA_8888:
64 return SkColorType::kRGBA_8888_SkColorType;
65 case PixelFormat::BGRA_8888:
66 return SkColorType::kBGRA_8888_SkColorType;
67 case PixelFormat::ALPHA_8:
68 return SkColorType::kAlpha_8_SkColorType;
69 case PixelFormat::RGBA_F16:
70 return SkColorType::kRGBA_F16_SkColorType;
71 case PixelFormat::UNKNOWN:
72 case PixelFormat::ARGB_8888:
73 case PixelFormat::RGB_888:
74 case PixelFormat::NV21:
75 case PixelFormat::NV12:
76 case PixelFormat::CMYK:
77 default:
78 return SkColorType::kUnknown_SkColorType;
79 }
80 }
81
MakeSkImageInfoFromPixelMap(const RefPtr<PixelMap> & pixmap)82 SkImageInfo MakeSkImageInfoFromPixelMap(const RefPtr<PixelMap>& pixmap)
83 {
84 SkColorType colorType = PixelFormatToSkColorType(pixmap);
85 SkAlphaType alphaType = AlphaTypeToSkAlphaType(pixmap);
86 sk_sp<SkColorSpace> colorSpace = ColorSpaceToSkColorSpace(pixmap);
87 return SkImageInfo::Make(pixmap->GetWidth(), pixmap->GetHeight(), colorType, alphaType, colorSpace);
88 }
89
GetWindow(int32_t containerId)90 const OHOS::sptr<OHOS::Rosen::Window> GetWindow(int32_t containerId)
91 {
92 auto container = AceEngine::Get().GetContainer(containerId);
93 if (containerId >= MIN_SUBCONTAINER_ID && containerId < MIN_PLUGIN_SUBCONTAINER_ID) {
94 auto subwindow = SubwindowManager::GetInstance()->GetSubwindow(
95 SubwindowManager::GetInstance()->GetParentContainerId(containerId));
96 CHECK_NULL_RETURN(subwindow, nullptr);
97 if (AceType::InstanceOf<SubwindowOhos>(subwindow)) {
98 auto subWindowOhos = AceType::DynamicCast<SubwindowOhos>(subwindow);
99 CHECK_NULL_RETURN(subWindowOhos, nullptr);
100 return subWindowOhos->GetSubWindow();
101 }
102 } else {
103 auto aceContainer = AceType::DynamicCast<Platform::AceContainer>(container);
104 if (aceContainer != nullptr) {
105 return OHOS::Rosen::Window::Find(aceContainer->GetWindowName());
106 }
107 return OHOS::Rosen::Window::GetTopWindowWithId(container->GetWindowId());
108 }
109 return nullptr;
110 }
111 } // namespace
112
113 constexpr static char RECNODE_SELFID[] = "selfId";
114 constexpr static char RECNODE_NODEID[] = "nodeID";
115 constexpr static char RECNODE_NAME[] = "value";
116 constexpr static char RECNODE_DEBUGLINE[] = "debugLine";
117 constexpr static char RECNODE_CHILDREN[] = "RSNode";
118 constexpr static char ARK_DEBUGGER_LIB_PATH[] = "libark_connect_inspector.z.so";
119 static constexpr char START_PERFORMANCE_CHECK_MESSAGE[] = "StartArkPerformanceCheck";
120 static constexpr char END_PERFORMANCE_CHECK_MESSAGE[] = "EndArkPerformanceCheck";
121
122 bool LayoutInspector::stateProfilerStatus_ = false;
123 bool LayoutInspector::layoutInspectorStatus_ = false;
124 bool LayoutInspector::isUseStageModel_ = false;
125 std::mutex LayoutInspector::recMutex_;
126 ProfilerStatusCallback LayoutInspector::jsStateProfilerStatusCallback_ = nullptr;
127 RsProfilerNodeMountCallback LayoutInspector::rsProfilerNodeMountCallback_ = nullptr;
128 const char PNG_TAG[] = "png";
129 NG::InspectorTreeMap LayoutInspector::recNodeInfos_;
130 std::once_flag LayoutInspector::loadFlag;
131 void* LayoutInspector::handlerConnectServerSo = nullptr;
132 LayoutInspector::SetArkUICallback LayoutInspector::setArkUICallback = nullptr;
133
SupportInspector()134 void LayoutInspector::SupportInspector()
135 {
136 auto container = Container::Current();
137 CHECK_NULL_VOID(container);
138 if (!layoutInspectorStatus_) {
139 return;
140 }
141 std::string treeJsonStr;
142 GetInspectorTreeJsonStr(treeJsonStr, ContainerScope::CurrentId());
143 if (treeJsonStr.empty()) {
144 return;
145 }
146 auto message = JsonUtil::Create(true);
147 GetSnapshotJson(ContainerScope::CurrentId(), message);
148 CHECK_NULL_VOID(message);
149
150 auto sendTask = [treeJsonStr, jsonSnapshotStr = message->ToString(), container]() {
151 if (container->IsUseStageModel()) {
152 WebSocketManager::SendInspector(treeJsonStr, jsonSnapshotStr);
153 } else {
154 OHOS::Ace::ConnectServerManager::Get().SendInspector(treeJsonStr, jsonSnapshotStr);
155 }
156 };
157 BackgroundTaskExecutor::GetInstance().PostTask(std::move(sendTask));
158 }
159
SetStatus(bool layoutInspectorStatus)160 void LayoutInspector::SetStatus(bool layoutInspectorStatus)
161 {
162 layoutInspectorStatus_ = layoutInspectorStatus;
163 }
164
TriggerJsStateProfilerStatusCallback(bool status)165 void LayoutInspector::TriggerJsStateProfilerStatusCallback(bool status)
166 {
167 if (jsStateProfilerStatusCallback_) {
168 stateProfilerStatus_ = status;
169 jsStateProfilerStatusCallback_(status);
170 }
171 }
172
SetJsStateProfilerStatusCallback(ProfilerStatusCallback && callback)173 void LayoutInspector::SetJsStateProfilerStatusCallback(ProfilerStatusCallback&& callback)
174 {
175 jsStateProfilerStatusCallback_ = callback;
176 }
177
GetStateProfilerStatus()178 bool LayoutInspector::GetStateProfilerStatus()
179 {
180 return stateProfilerStatus_;
181 }
182
GetRsProfilerNodeMountCallback()183 RsProfilerNodeMountCallback LayoutInspector::GetRsProfilerNodeMountCallback()
184 {
185 return rsProfilerNodeMountCallback_;
186 }
187
SetRsProfilerNodeMountCallback(RsProfilerNodeMountCallback && callback)188 void LayoutInspector::SetRsProfilerNodeMountCallback(RsProfilerNodeMountCallback&& callback)
189 {
190 rsProfilerNodeMountCallback_ = callback;
191 }
192
SendMessage(const std::string & message)193 void LayoutInspector::SendMessage(const std::string& message)
194 {
195 WebSocketManager::SendMessage(message);
196 }
197
SetStateProfilerStatus(bool status)198 void LayoutInspector::SetStateProfilerStatus(bool status)
199 {
200 auto taskExecutor = Container::CurrentTaskExecutorSafely();
201 CHECK_NULL_VOID(taskExecutor);
202 auto task = [status]() { LayoutInspector::TriggerJsStateProfilerStatusCallback(status); };
203 taskExecutor->PostTask(std::move(task), TaskExecutor::TaskType::UI, "ArkUISetStateProfilerStatus");
204 }
205
ConnectServerCallback()206 void LayoutInspector::ConnectServerCallback()
207 {
208 TAG_LOGD(AceLogTag::ACE_LAYOUT_INSPECTOR, "connect server callback isStage:%{public}d", isUseStageModel_);
209 if (isUseStageModel_) {
210 TAG_LOGD(AceLogTag::ACE_LAYOUT_INSPECTOR, "connect server, reset callback.");
211 WebSocketManager::SetRecordCallback(LayoutInspector::HandleStartRecord, LayoutInspector::HandleStopRecord);
212 }
213 }
214
SetCallback(int32_t instanceId)215 void LayoutInspector::SetCallback(int32_t instanceId)
216 {
217 TAG_LOGD(AceLogTag::ACE_LAYOUT_INSPECTOR, "InstanceId:%{public}d", instanceId);
218 auto container = AceEngine::Get().GetContainer(instanceId);
219 CHECK_NULL_VOID(container);
220 if (container->IsUseStageModel()) {
221 WebSocketManager::SetProfilerCallBack([](bool status) { return SetStateProfilerStatus(status); });
222 WebSocketManager::SetSwitchCallback(
223 [](int32_t containerId) { return CreateLayoutInfo(containerId); }, instanceId);
224 WebSocketManager::SetRecordCallback(LayoutInspector::HandleStartRecord, LayoutInspector::HandleStopRecord);
225 WebSocketManager::RegisterConnectServerCallback(LayoutInspector::ConnectServerCallback);
226 isUseStageModel_ = true;
227 RegisterConnectCallback();
228 } else {
229 OHOS::Ace::ConnectServerManager::Get().SetLayoutInspectorCallback(
230 [](int32_t containerId) { return CreateLayoutInfo(containerId); });
231 isUseStageModel_ = false;
232 }
233
234 SendInstanceMessageCallBack sendInstanceMessageCallBack = [](int32_t id) {
235 WebSocketManager::SetProfilerCallBack(
236 [](bool status) { return SetStateProfilerStatus(status); });
237 WebSocketManager::SetSwitchCallback([](int32_t containerId) { return CreateLayoutInfo(containerId); }, id);
238 };
239 WebSocketManager::RegisterSendInstanceMessageCallback(sendInstanceMessageCallBack);
240 }
241
CreateContainerLayoutInfo(RefPtr<Container> & container)242 void LayoutInspector::CreateContainerLayoutInfo(RefPtr<Container>& container)
243 {
244 CHECK_NULL_VOID(container);
245 if (container->IsDynamicRender()) {
246 container = Container::CurrentSafely();
247 CHECK_NULL_VOID(container);
248 }
249 int32_t containerId = container->GetInstanceId();
250 ContainerScope socpe(containerId);
251 auto context = PipelineContext::GetCurrentContext();
252 CHECK_NULL_VOID(context);
253 auto getInspectorTask = [container, containerId]() {
254 std::string treeJson;
255 GetInspectorTreeJsonStr(treeJson, containerId);
256 auto message = JsonUtil::Create(true);
257 GetSnapshotJson(containerId, message);
258 CHECK_NULL_VOID(message);
259 auto sendResultTask = [treeJsonStr = std::move(treeJson), jsonSnapshotStr = message->ToString(), container]() {
260 if (container->IsUseStageModel()) {
261 WebSocketManager::SendInspector(treeJsonStr, jsonSnapshotStr);
262 } else {
263 OHOS::Ace::ConnectServerManager::Get().SendInspector(treeJsonStr, jsonSnapshotStr);
264 }
265 };
266 BackgroundTaskExecutor::GetInstance().PostTask(std::move(sendResultTask));
267 };
268 context->GetTaskExecutor()->PostTask(
269 std::move(getInspectorTask), TaskExecutor::TaskType::UI, "ArkUIGetInspectorTreeJson");
270 }
271
CreateLayoutInfo(int32_t containerId)272 void LayoutInspector::CreateLayoutInfo(int32_t containerId)
273 {
274 auto container = Container::GetFoucsed();
275 return CreateContainerLayoutInfo(container);
276 }
277
CreateLayoutInfoByWinId(uint32_t windId)278 void LayoutInspector::CreateLayoutInfoByWinId(uint32_t windId)
279 {
280 auto container = Container::GetByWindowId(windId);
281 if (container) {
282 TAG_LOGD(AceLogTag::ACE_LAYOUT_INSPECTOR, "start get container %{public}d info", container->GetInstanceId());
283 }
284 return CreateContainerLayoutInfo(container);
285 }
286
GetInspectorTreeJsonStr(std::string & treeJsonStr,int32_t containerId)287 void LayoutInspector::GetInspectorTreeJsonStr(std::string& treeJsonStr, int32_t containerId)
288 {
289 auto container = AceEngine::Get().GetContainer(containerId);
290 CHECK_NULL_VOID(container);
291 #ifdef NG_BUILD
292 treeJsonStr = NG::Inspector::GetInspector(true);
293 #else
294 if (container->IsUseNewPipeline()) {
295 if (containerId >= MIN_SUBCONTAINER_ID && containerId < MIN_PLUGIN_SUBCONTAINER_ID) {
296 treeJsonStr = NG::Inspector::GetSubWindowInspector(true);
297 } else {
298 treeJsonStr = NG::Inspector::GetInspector(true);
299 }
300 } else {
301 auto pipelineContext = AceType::DynamicCast<PipelineContext>(container->GetPipelineContext());
302 CHECK_NULL_VOID(pipelineContext);
303 treeJsonStr = V2::Inspector::GetInspectorTree(pipelineContext, true);
304 }
305 #endif
306 }
307
GetSnapshotJson(int32_t containerId,std::unique_ptr<JsonValue> & message)308 void LayoutInspector::GetSnapshotJson(int32_t containerId, std::unique_ptr<JsonValue>& message)
309 {
310 auto container = AceEngine::Get().GetContainer(containerId);
311 CHECK_NULL_VOID(container);
312 OHOS::sptr<OHOS::Rosen::Window> window = GetWindow(containerId);
313 CHECK_NULL_VOID(window);
314 auto pixelMap = window->Snapshot();
315 CHECK_NULL_VOID(pixelMap);
316 auto acePixelMap = AceType::MakeRefPtr<PixelMapOhos>(pixelMap);
317 CHECK_NULL_VOID(acePixelMap);
318 auto imageInfo = MakeSkImageInfoFromPixelMap(acePixelMap);
319 SkPixmap imagePixmap(
320 imageInfo, reinterpret_cast<const void*>(acePixelMap->GetPixels()), acePixelMap->GetRowBytes());
321 sk_sp<SkImage> image;
322 image = SkImage::MakeFromRaster(imagePixmap, &PixelMap::ReleaseProc, PixelMap::GetReleaseContext(acePixelMap));
323 CHECK_NULL_VOID(image);
324 auto data = image->encodeToData(SkEncodedImageFormat::kPNG, 100);
325 CHECK_NULL_VOID(data);
326 auto defaultDisplay = Rosen::DisplayManager::GetInstance().GetDefaultDisplay();
327 CHECK_NULL_VOID(defaultDisplay);
328 auto deviceDpi = defaultDisplay->GetDpi();
329 auto deviceWidth = defaultDisplay->GetWidth();
330 auto deviceHeight = defaultDisplay->GetHeight();
331 message->Put("type", "snapShot");
332 message->Put("format", PNG_TAG);
333 message->Put("width", (*pixelMap).GetWidth());
334 message->Put("height", (*pixelMap).GetHeight());
335 message->Put("posX", container->GetViewPosX());
336 message->Put("posY", container->GetViewPosY());
337 message->Put("deviceWidth", deviceWidth);
338 message->Put("deviceHeight", deviceHeight);
339 message->Put("deviceDpi", deviceDpi);
340 int32_t encodeLength = static_cast<int32_t>(SkBase64::Encode(data->data(), data->size(), nullptr));
341 message->Put("size", data->size());
342 SkString info(encodeLength);
343 SkBase64::Encode(data->data(), data->size(), info.writable_str());
344 message->Put("pixelMapBase64", info.c_str());
345 }
346
RegisterConnectCallback()347 void LayoutInspector::RegisterConnectCallback()
348 {
349 std::call_once(loadFlag, []() {
350 handlerConnectServerSo = dlopen(ARK_DEBUGGER_LIB_PATH, RTLD_NOLOAD | RTLD_NOW);
351 if (handlerConnectServerSo == nullptr) {
352 handlerConnectServerSo = dlopen(ARK_DEBUGGER_LIB_PATH, RTLD_NOW);
353 if (handlerConnectServerSo == nullptr) {
354 TAG_LOGE(AceLogTag::ACE_LAYOUT_INSPECTOR, "null handlerConnectServerSo: %{public}s", dlerror());
355 return;
356 }
357 }
358
359 setArkUICallback = reinterpret_cast<SetArkUICallback>(dlsym(handlerConnectServerSo, "SetArkUICallback"));
360 if (setArkUICallback == nullptr) {
361 TAG_LOGE(AceLogTag::ACE_LAYOUT_INSPECTOR, "null setArkUICallback: %{public}s", dlerror());
362 return;
363 }
364 });
365
366 if (setArkUICallback != nullptr) {
367 setArkUICallback([](const char* message) { ProcessMessages(message); });
368 }
369 }
370
ProcessMessages(const std::string & message)371 void LayoutInspector::ProcessMessages(const std::string& message)
372 {
373 if (message.find(START_PERFORMANCE_CHECK_MESSAGE, 0) != std::string::npos) {
374 TAG_LOGI(AceLogTag::ACE_LAYOUT_INSPECTOR, "performance check start");
375 AceChecker::SetPerformanceCheckStatus(true, message);
376 } else if (message.find(END_PERFORMANCE_CHECK_MESSAGE, 0) != std::string::npos) {
377 TAG_LOGI(AceLogTag::ACE_LAYOUT_INSPECTOR, "performance check end");
378 AceChecker::SetPerformanceCheckStatus(false, message);
379 }
380 uint32_t windowId = NG::Inspector::ParseWindowIdFromMsg(message);
381 if (windowId == OHOS::Ace::NG::INVALID_WINDOW_ID) {
382 TAG_LOGE(AceLogTag::ACE_LAYOUT_INSPECTOR, "input message: %{public}s", message.c_str());
383 return;
384 }
385 CreateLayoutInfoByWinId(windowId);
386 }
387
HandleStopRecord()388 void LayoutInspector::HandleStopRecord()
389 {
390 std::unique_lock<std::mutex> lock(recMutex_);
391 SetRsProfilerNodeMountCallback(nullptr);
392 auto jsonRoot = JsonUtil::Create(true);
393 auto jsonNodeArray = JsonUtil::CreateArray(true);
394 for (auto& uiNode : recNodeInfos_) {
395 if (uiNode.second != nullptr) {
396 auto jsonNode = JsonUtil::Create(true);
397 jsonNode->Put(RECNODE_NODEID, std::to_string(uiNode.second->GetSelfId()).c_str());
398 jsonNode->Put(RECNODE_SELFID, uiNode.second->GetNodeId());
399 jsonNode->Put(RECNODE_NAME, uiNode.second->GetName().c_str());
400 jsonNode->Put(RECNODE_DEBUGLINE, uiNode.second->GetDebugLine().c_str());
401 jsonNodeArray->PutRef(std::move(jsonNode));
402 }
403 }
404 recNodeInfos_.clear();
405 lock.unlock();
406 if (jsonNodeArray->GetArraySize()) {
407 jsonRoot->PutRef(RECNODE_CHILDREN, std::move(jsonNodeArray));
408 }
409 std::string arrayJsonStr = jsonRoot->ToString();
410 auto sendResultTask = [arrayJsonStr]() {
411 WebSocketManager::SetRecordResults(arrayJsonStr);
412 };
413 BackgroundTaskExecutor::GetInstance().PostTask(std::move(sendResultTask));
414 }
415
HandleStartRecord()416 void LayoutInspector::HandleStartRecord()
417 {
418 // regist inner callback function
419 std::unique_lock<std::mutex> lock(recMutex_);
420 SetRsProfilerNodeMountCallback(LayoutInspector::HandleInnerCallback);
421 lock.unlock();
422 auto container = Container::GetFoucsed();
423 CHECK_NULL_VOID(container);
424 if (container->IsDynamicRender()) {
425 container = Container::CurrentSafely();
426 CHECK_NULL_VOID(container);
427 }
428 auto containerId = container->GetInstanceId();
429 ContainerScope socpe(containerId);
430 auto context = PipelineContext::GetCurrentContext();
431 CHECK_NULL_VOID(context);
432 auto startRecordTask = []() {
433 std::lock_guard<std::mutex> lock(LayoutInspector::recMutex_);
434 NG::InspectorTreeMap recTreeNodes;
435 NG::InspectorTreeMap offScreenTreeNodes;
436 NG::Inspector::GetRecordAllPagesNodes(recTreeNodes);
437 TAG_LOGD(AceLogTag::ACE_LAYOUT_INSPECTOR, "Get nodes size:%{public}zu", recTreeNodes.size());
438 NG::Inspector::GetOffScreenTreeNodes(offScreenTreeNodes);
439 TAG_LOGD(AceLogTag::ACE_LAYOUT_INSPECTOR, "Get offscreen nodes size:%{public}zu", offScreenTreeNodes.size());
440 LayoutInspector::recNodeInfos_.swap(recTreeNodes);
441 for (auto& item : offScreenTreeNodes) {
442 recNodeInfos_.emplace(item);
443 }
444 };
445 context->GetTaskExecutor()->PostTask(
446 std::move(startRecordTask), TaskExecutor::TaskType::UI, "ArkUIGetInspectorTree");
447 }
448
HandleInnerCallback(FrameNodeInfo node)449 void LayoutInspector::HandleInnerCallback(FrameNodeInfo node)
450 {
451 // convert FrameNodeInfo --> recNode
452 TAG_LOGD(AceLogTag::ACE_LAYOUT_INSPECTOR,
453 "FrameNodeInfo:selfid:%{public}" PRIu64 ",nodid:%{public}d,type:%{public}s,debugline:%{public}s",
454 node.rsNodeId, node.frameNodeId, node.nodeType.c_str(), node.debugline.c_str());
455 auto recNode = AceType::MakeRefPtr<NG::RecNode>();
456 CHECK_NULL_VOID(recNode);
457 recNode->SetSelfId(node.rsNodeId);
458 recNode->SetNodeId(node.frameNodeId);
459 recNode->SetName(node.nodeType);
460 recNode->SetDebugLine(node.debugline);
461 std::lock_guard<std::mutex> lock(recMutex_);
462 recNodeInfos_.emplace(node.rsNodeId, recNode);
463 }
464 } // namespace OHOS::Ace
465