• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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