1 /*
2 * Copyright (c) 2023 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 "session_manager.h"
17
18 #include <iservice_registry.h>
19 #include <system_ability_definition.h>
20
21 #include "session_manager_service_recover_interface.h"
22 #include "singleton_delegator.h"
23 #include "window_manager_hilog.h"
24
25 namespace OHOS::Rosen {
26 namespace {
27 constexpr HiviewDFX::HiLogLabel LABEL = { LOG_CORE, HILOG_DOMAIN_DISPLAY, "SessionManager" };
28 }
29
30 class SessionManagerServiceRecoverListener : public IRemoteStub<ISessionManagerServiceRecoverListener> {
31 public:
32 explicit SessionManagerServiceRecoverListener() = default;
33
OnRemoteRequest(uint32_t code,MessageParcel & data,MessageParcel & reply,MessageOption & option)34 virtual int32_t OnRemoteRequest(
35 uint32_t code, MessageParcel &data, MessageParcel &reply, MessageOption &option) override
36 {
37 if (data.ReadInterfaceToken() != GetDescriptor()) {
38 WLOGFE("InterfaceToken check failed");
39 return -1;
40 }
41 auto msgId = static_cast<SessionManagerServiceRecoverMessage>(code);
42 switch (msgId) {
43 case SessionManagerServiceRecoverMessage::TRANS_ID_ON_SESSION_MANAGER_SERVICE_RECOVER: {
44 auto sessionManagerService = data.ReadRemoteObject();
45 OnSessionManagerServiceRecover(sessionManagerService);
46 break;
47 }
48 case SessionManagerServiceRecoverMessage::TRANS_ID_ON_WMS_CONNECTION_CHANGED: {
49 int32_t userId = data.ReadInt32();
50 int32_t screenId = data.ReadInt32();
51 bool isConnected = data.ReadBool();
52 OnWMSConnectionChanged(userId, screenId, isConnected);
53 break;
54 }
55 default:
56 WLOGFW("unknown transaction code %{public}d", code);
57 return IPCObjectStub::OnRemoteRequest(code, data, reply, option);
58 }
59 return 0;
60 }
61
OnSessionManagerServiceRecover(const sptr<IRemoteObject> & sessionManagerService)62 void OnSessionManagerServiceRecover(const sptr<IRemoteObject>& sessionManagerService) override
63 {
64 SessionManager::GetInstance().Clear();
65 SessionManager::GetInstance().ClearSessionManagerProxy();
66
67 auto sms = iface_cast<ISessionManagerService>(sessionManagerService);
68 SessionManager::GetInstance().RecoverSessionManagerService(sms);
69 }
70
OnWMSConnectionChanged(int32_t userId,int32_t screenId,bool isConnected)71 void OnWMSConnectionChanged(int32_t userId, int32_t screenId, bool isConnected) override
72 {
73 SessionManager::GetInstance().OnWMSConnectionChanged(userId, screenId, isConnected);
74 }
75 };
76
WM_IMPLEMENT_SINGLE_INSTANCE(SessionManager)77 WM_IMPLEMENT_SINGLE_INSTANCE(SessionManager)
78
79 SessionManager::~SessionManager()
80 {
81 WLOGFI("SessionManager destory!");
82 std::lock_guard<std::recursive_mutex> lock(mutex_);
83 destroyed_ = true;
84 if (mockSessionManagerServiceProxy_ != nullptr) {
85 mockSessionManagerServiceProxy_->UnregisterSMSRecoverListener();
86 mockSessionManagerServiceProxy_ = nullptr;
87 }
88 }
89
OnWMSConnectionChanged(int32_t userId,int32_t screenId,bool isConnected)90 void SessionManager::OnWMSConnectionChanged(int32_t userId, int32_t screenId, bool isConnected)
91 {
92 WLOGFD("OnWMSConnectionChanged");
93 isWMSConnected_ = isConnected;
94 currentUserId_ = userId;
95 currentScreenId_ = screenId;
96 if (wmsConnectionChangedFunc_ != nullptr) {
97 WLOGFI("WMS connection changed with userId=%{public}d, screenId=%{public}d, isConnected=%{public}d",
98 userId, screenId, isConnected);
99 wmsConnectionChangedFunc_(userId, screenId, isConnected);
100 }
101 }
102
ClearSessionManagerProxy()103 void SessionManager::ClearSessionManagerProxy()
104 {
105 WLOGFI("ClearSessionManagerProxy enter!");
106 std::lock_guard<std::recursive_mutex> lock(mutex_);
107 if (destroyed_) {
108 WLOGFE("Already destroyed");
109 return;
110 }
111
112 if (sessionManagerServiceProxy_ != nullptr) {
113 int refCount = sessionManagerServiceProxy_->GetSptrRefCount();
114 WLOGFI("sessionManagerServiceProxy_ GetSptrRefCount : %{public}d", refCount);
115 sessionManagerServiceProxy_ = nullptr;
116 }
117 sceneSessionManagerProxy_ = nullptr;
118 }
119
120 __attribute__((no_sanitize("cfi")))
GetSceneSessionManagerProxy()121 sptr<ISceneSessionManager> SessionManager::GetSceneSessionManagerProxy()
122 {
123 std::lock_guard<std::recursive_mutex> lock(mutex_);
124 InitSessionManagerServiceProxy();
125 InitSceneSessionManagerProxy();
126 return sceneSessionManagerProxy_;
127 }
128
InitSessionManagerServiceProxy()129 void SessionManager::InitSessionManagerServiceProxy()
130 {
131 if (sessionManagerServiceProxy_) {
132 return;
133 }
134 sptr<ISystemAbilityManager> systemAbilityManager =
135 SystemAbilityManagerClient::GetInstance().GetSystemAbilityManager();
136 if (!systemAbilityManager) {
137 WLOGFE("Failed to get system ability mgr.");
138 return;
139 }
140 sptr<IRemoteObject> remoteObject = systemAbilityManager->GetSystemAbility(WINDOW_MANAGER_SERVICE_ID);
141 if (!remoteObject) {
142 WLOGFI("Remote object is nullptr");
143 return;
144 }
145 mockSessionManagerServiceProxy_ = iface_cast<IMockSessionManagerInterface>(remoteObject);
146 if (!mockSessionManagerServiceProxy_) {
147 WLOGFW("Get mock session manager service proxy failed, nullptr");
148 return;
149 }
150
151 if (!isRecoverListenerRegistered_) {
152 isRecoverListenerRegistered_ = true;
153 WLOGFI("[WMSRecover] Register recover listener");
154 smsRecoverListener_ = new SessionManagerServiceRecoverListener();
155 mockSessionManagerServiceProxy_->RegisterSMSRecoverListener(smsRecoverListener_);
156 }
157
158 sptr<IRemoteObject> remoteObject2 = mockSessionManagerServiceProxy_->GetSessionManagerService();
159 if (!remoteObject2) {
160 WLOGFE("Remote object2 is nullptr");
161 return;
162 }
163 sessionManagerServiceProxy_ = iface_cast<ISessionManagerService>(remoteObject2);
164 if (!sessionManagerServiceProxy_) {
165 WLOGFE("sessionManagerServiceProxy_ is nullptr");
166 }
167 }
168
169 __attribute__((no_sanitize("cfi")))
InitSceneSessionManagerProxy()170 void SessionManager::InitSceneSessionManagerProxy()
171 {
172 if (sceneSessionManagerProxy_) {
173 return;
174 }
175 if (!sessionManagerServiceProxy_) {
176 WLOGFE("sessionManagerServiceProxy_ is nullptr");
177 return;
178 }
179
180 sptr<IRemoteObject> remoteObject = sessionManagerServiceProxy_->GetSceneSessionManager();
181 if (!remoteObject) {
182 WLOGFW("Get scene session manager proxy failed, scene session manager service is null");
183 return;
184 }
185 sceneSessionManagerProxy_ = iface_cast<ISceneSessionManager>(remoteObject);
186 if (sceneSessionManagerProxy_) {
187 ssmDeath_ = new (std::nothrow) SSMDeathRecipient();
188 if (!ssmDeath_) {
189 WLOGFE("Failed to create death Recipient ptr WMSDeathRecipient");
190 return;
191 }
192 if (remoteObject->IsProxyObject() && !remoteObject->AddDeathRecipient(ssmDeath_)) {
193 WLOGFE("Failed to add death recipient");
194 return;
195 }
196 }
197 if (!sceneSessionManagerProxy_) {
198 WLOGFW("Get scene session manager proxy failed, nullptr");
199 }
200 }
201
RegisterWindowManagerRecoverCallbackFunc(const WindowManagerRecoverCallbackFunc & callbackFunc)202 void SessionManager::RegisterWindowManagerRecoverCallbackFunc(const WindowManagerRecoverCallbackFunc& callbackFunc)
203 {
204 std::lock_guard<std::recursive_mutex> lock(recoverMutex_);
205 windowManagerRecoverFunc_ = callbackFunc;
206 }
207
RecoverSessionManagerService(const sptr<ISessionManagerService> & sessionManagerService)208 void SessionManager::RecoverSessionManagerService(const sptr<ISessionManagerService>& sessionManagerService)
209 {
210 {
211 std::lock_guard<std::recursive_mutex> lock(mutex_);
212 sessionManagerServiceProxy_ = sessionManagerService;
213 }
214
215 {
216 std::lock_guard<std::recursive_mutex> lock(recoverMutex_);
217 WLOGFI("[WMSRecover] Run recover");
218 if (windowManagerRecoverFunc_ != nullptr) {
219 WLOGFD("[WMSRecover] windowManagerRecover");
220 windowManagerRecoverFunc_();
221 }
222 }
223 }
224
Clear()225 void SessionManager::Clear()
226 {
227 std::lock_guard<std::recursive_mutex> lock(mutex_);
228 if ((sceneSessionManagerProxy_ != nullptr) && (sceneSessionManagerProxy_->AsObject() != nullptr)) {
229 sceneSessionManagerProxy_->AsObject()->RemoveDeathRecipient(ssmDeath_);
230 }
231 }
232
RegisterWMSConnectionChangedListener(const WMSConnectionChangedCallbackFunc & callbackFunc)233 void SessionManager::RegisterWMSConnectionChangedListener(const WMSConnectionChangedCallbackFunc& callbackFunc)
234 {
235 WLOGFI("RegisterWMSConnectionChangedListener in");
236 wmsConnectionChangedFunc_ = callbackFunc;
237 if (isWMSConnected_) {
238 OnWMSConnectionChanged(currentUserId_, currentScreenId_, true);
239 }
240 }
241
OnRemoteDied(const wptr<IRemoteObject> & wptrDeath)242 void SSMDeathRecipient::OnRemoteDied(const wptr<IRemoteObject>& wptrDeath)
243 {
244 if (wptrDeath == nullptr) {
245 WLOGFE("SSMDeathRecipient wptrDeath is null");
246 return;
247 }
248
249 sptr<IRemoteObject> object = wptrDeath.promote();
250 if (!object) {
251 WLOGFE("SSMDeathRecipient object is null");
252 return;
253 }
254 WLOGI("ssm OnRemoteDied");
255 SessionManager::GetInstance().Clear();
256 SessionManager::GetInstance().ClearSessionManagerProxy();
257 }
258 } // namespace OHOS::Rosen
259