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 "switch_subscriber_handler.h"
17
18 #include "bytrace_adapter.h"
19 #include "define_multimodal.h"
20 #include "error_multimodal.h"
21 #include "input_event_data_transformation.h"
22 #include "input_event_handler.h"
23 #include "net_packet.h"
24 #include "proto.h"
25 #include "util_ex.h"
26 #include "dfx_hisysevent.h"
27
28 namespace OHOS {
29 namespace MMI {
30 namespace {
31 constexpr OHOS::HiviewDFX::HiLogLabel LABEL = { LOG_CORE, MMI_LOG_DOMAIN, "SwitchSubscriberHandler" };
32 } // namespace
33
34 #ifdef OHOS_BUILD_ENABLE_KEYBOARD
HandleKeyEvent(const std::shared_ptr<KeyEvent> keyEvent)35 void SwitchSubscriberHandler::HandleKeyEvent(const std::shared_ptr<KeyEvent> keyEvent)
36 {
37 CHKPV(keyEvent);
38 CHKPV(nextHandler_);
39 nextHandler_->HandleKeyEvent(keyEvent);
40 }
41 #endif // OHOS_BUILD_ENABLE_KEYBOARD
42
43 #ifdef OHOS_BUILD_ENABLE_POINTER
HandlePointerEvent(const std::shared_ptr<PointerEvent> pointerEvent)44 void SwitchSubscriberHandler::HandlePointerEvent(const std::shared_ptr<PointerEvent> pointerEvent)
45 {
46 CHKPV(pointerEvent);
47 CHKPV(nextHandler_);
48 nextHandler_->HandlePointerEvent(pointerEvent);
49 }
50 #endif // OHOS_BUILD_ENABLE_POINTER
51
52 #ifdef OHOS_BUILD_ENABLE_TOUCH
HandleTouchEvent(const std::shared_ptr<PointerEvent> pointerEvent)53 void SwitchSubscriberHandler::HandleTouchEvent(const std::shared_ptr<PointerEvent> pointerEvent)
54 {
55 CHKPV(pointerEvent);
56 CHKPV(nextHandler_);
57 nextHandler_->HandleTouchEvent(pointerEvent);
58 }
59 #endif // OHOS_BUILD_ENABLE_TOUCH
60
61 #ifdef OHOS_BUILD_ENABLE_SWITCH
HandleSwitchEvent(const std::shared_ptr<SwitchEvent> switchEvent)62 void SwitchSubscriberHandler::HandleSwitchEvent(const std::shared_ptr<SwitchEvent> switchEvent)
63 {
64 CHKPV(switchEvent);
65 if (OnSubscribeSwitchEvent(switchEvent)) {
66 MMI_HILOGD("Subscribe switchEvent filter success. switchValue:%{public}d", switchEvent->GetSwitchValue());
67 return;
68 }
69 CHKPV(nextHandler_);
70 nextHandler_->HandleSwitchEvent(switchEvent);
71 }
72 #endif // OHOS_BUILD_ENABLE_SWITCH
73
SubscribeSwitchEvent(SessionPtr sess,int32_t subscribeId)74 int32_t SwitchSubscriberHandler::SubscribeSwitchEvent(SessionPtr sess, int32_t subscribeId)
75 {
76 CALL_INFO_TRACE;
77 if (subscribeId < 0) {
78 MMI_HILOGE("Invalid subscribeId");
79 return RET_ERR;
80 }
81 CHKPR(sess, ERROR_NULL_POINTER);
82
83 MMI_HILOGD("subscribeId:%{public}d", subscribeId);
84 auto subscriber = std::make_shared<Subscriber>(subscribeId, sess);
85 InsertSubScriber(std::move(subscriber));
86 InitSessionDeleteCallback();
87 return RET_OK;
88 }
89
UnsubscribeSwitchEvent(SessionPtr sess,int32_t subscribeId)90 int32_t SwitchSubscriberHandler::UnsubscribeSwitchEvent(SessionPtr sess, int32_t subscribeId)
91 {
92 CALL_INFO_TRACE;
93 MMI_HILOGD("subscribeId:%{public}d", subscribeId);
94 for (auto it = subscribers_.begin(); it != subscribers_.end(); ++it) {
95 if ((*it)->id_ == subscribeId && (*it)->sess_ == sess) {
96 subscribers_.erase(it);
97 return RET_OK;
98 }
99 }
100 MMI_HILOGE("UnsubscribeSwitchEvent failed with %{public}d", subscribeId);
101 return RET_ERR;
102 }
103
OnSubscribeSwitchEvent(std::shared_ptr<SwitchEvent> switchEvent)104 bool SwitchSubscriberHandler::OnSubscribeSwitchEvent(std::shared_ptr<SwitchEvent> switchEvent)
105 {
106 CHKPF(switchEvent);
107 MMI_HILOGD("switchValue:%{public}d", switchEvent->GetSwitchValue());
108 DfxHisysevent::OnLidSwitchChanged(switchEvent->GetSwitchValue());
109
110 bool handled = false;
111 for (const auto &subscriber : subscribers_) {
112 NotifySubscriber(switchEvent, subscriber);
113 handled = true;
114 }
115 MMI_HILOGD("%{public}s", handled ? "true" : "false");
116 return handled;
117 }
118
InsertSubScriber(std::shared_ptr<Subscriber> subs)119 void SwitchSubscriberHandler::InsertSubScriber(std::shared_ptr<Subscriber> subs)
120 {
121 CALL_DEBUG_ENTER;
122 CHKPV(subs);
123 for (auto it = subscribers_.begin(); it != subscribers_.end(); ++it) {
124 if (subs->sess_ != nullptr && (*it)->id_ == subs->id_ && (*it)->sess_ == subs->sess_) {
125 MMI_HILOGW("Repeat registration id:%{public}d desc:%{public}s",
126 subs->id_, subs->sess_->GetDescript().c_str());
127 return;
128 }
129 }
130 subscribers_.push_back(subs);
131 }
132
OnSessionDelete(SessionPtr sess)133 void SwitchSubscriberHandler::OnSessionDelete(SessionPtr sess)
134 {
135 CALL_DEBUG_ENTER;
136 CHKPV(sess);
137 for (auto it = subscribers_.begin(); it != subscribers_.end();) {
138 if ((*it)->sess_ == sess) {
139 subscribers_.erase(it++);
140 continue;
141 }
142 ++it;
143 }
144 }
145
NotifySubscriber(std::shared_ptr<SwitchEvent> switchEvent,const std::shared_ptr<Subscriber> & subscriber)146 void SwitchSubscriberHandler::NotifySubscriber(std::shared_ptr<SwitchEvent> switchEvent,
147 const std::shared_ptr<Subscriber> &subscriber)
148 {
149 CALL_DEBUG_ENTER;
150 CHKPV(switchEvent);
151 CHKPV(subscriber);
152 auto udsServerPtr = InputHandler->GetUDSServer();
153 CHKPV(udsServerPtr);
154 NetPacket pkt(MmiMessageId::ON_SUBSCRIBE_SWITCH);
155 InputEventDataTransformation::SwitchEventToNetPacket(switchEvent, pkt);
156 if (subscriber->sess_ == nullptr) {
157 MMI_HILOGE("subscriber's sess is null");
158 return;
159 }
160 int32_t fd = subscriber->sess_->GetFd();
161 pkt << fd << subscriber->id_;
162 MMI_HILOGI("Notify subscriber id:%{public}d, switchValue:%{public}d, pid:%{public}d",
163 subscriber->id_, switchEvent->GetSwitchValue(), subscriber->sess_->GetPid());
164 if (pkt.ChkRWError()) {
165 MMI_HILOGE("Packet write dispatch subscriber failed");
166 return;
167 }
168 if (!udsServerPtr->SendMsg(fd, pkt)) {
169 MMI_HILOGE("Leave, server dispatch subscriber failed");
170 }
171 }
172
InitSessionDeleteCallback()173 bool SwitchSubscriberHandler::InitSessionDeleteCallback()
174 {
175 CALL_DEBUG_ENTER;
176 if (callbackInitialized_) {
177 MMI_HILOGD("Session delete callback has already been initialized");
178 return true;
179 }
180 auto udsServerPtr = InputHandler->GetUDSServer();
181 CHKPF(udsServerPtr);
182 std::function<void(SessionPtr)> callback =
183 std::bind(&SwitchSubscriberHandler::OnSessionDelete, this, std::placeholders::_1);
184 udsServerPtr->AddSessionDeletedCallback(callback);
185 callbackInitialized_ = true;
186 return true;
187 }
188
Dump(int32_t fd,const std::vector<std::string> & args)189 void SwitchSubscriberHandler::Dump(int32_t fd, const std::vector<std::string> &args)
190 {
191 CALL_DEBUG_ENTER;
192 mprintf(fd, "Subscriber information:\t");
193 mprintf(fd, "subscribers: count=%d", subscribers_.size());
194 for (const auto &item : subscribers_) {
195 std::shared_ptr<Subscriber> subscriber = item;
196 CHKPV(subscriber);
197 SessionPtr session = item->sess_;
198 CHKPV(session);
199 mprintf(fd, "subscriber id:%d | Pid:%d | Uid:%d | Fd:%d\t",
200 subscriber->id_, session->GetPid(), session->GetUid(), session->GetFd());
201 }
202 }
203 } // namespace MMI
204 } // namespace OHOS
205