1 /*
2  * Copyright (c) 2021-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 <cassert>
17 #include <chrono>
18 
19 #include "axis_event.h"
20 #include "event_log_helper.h"
21 #include "input_event.h"
22 #include "key_event.h"
23 #include "mmi_log.h"
24 #include "pointer_event.h"
25 
26 #undef MMI_LOG_TAG
27 #define MMI_LOG_TAG "InputEvent"
28 
29 namespace OHOS {
30 namespace MMI {
31 namespace {
32 int64_t g_nextEventId = 1;
33 constexpr uint32_t DATA_LENGTH_LIMIT { 1024 }; // 1024: max length
34 } // namespace
35 
36 std::string EventLogHelper::userType_ = "";
37 std::once_flag EventLogHelper::betaFlag_;
38 
InputEvent(int32_t eventType)39 InputEvent::InputEvent(int32_t eventType) : eventType_(eventType)
40 {
41     Reset();
42 }
43 
InputEvent(const InputEvent& other)44 InputEvent::InputEvent(const InputEvent& other)
45     : eventType_(other.eventType_), id_(other.id_), actionTime_(other.actionTime_),
46       sensorInputTime_(other.sensorInputTime_), action_(other.action_), actionStartTime_(other.actionStartTime_),
47       deviceId_(other.deviceId_), targetDisplayId_(other.targetDisplayId_),
48       targetWindowId_(other.targetWindowId_), agentWindowId_(other.agentWindowId_),
49       bitwise_(other.bitwise_), markEnabled_(other.markEnabled_),
50       extraData_(other.extraData_), extraDataLength_(other.extraDataLength_) {}
51 
~InputEvent()52 InputEvent::~InputEvent() {}
53 
Reset()54 void InputEvent::Reset()
55 {
56     struct timespec ts = { 0, 0 };
57     if (clock_gettime(CLOCK_MONOTONIC, &ts) != 0) {
58         actionTime_ = 0;
59     }
60     id_ = -1;
61     if (!AddInt64(ts.tv_sec * 1000000, ts.tv_nsec / 1000, actionTime_)) {
62         MMI_HILOGE("The addition of actionTime_ overflows");
63         return;
64     }
65     action_ = ACTION_UNKNOWN;
66     actionStartTime_ = actionTime_;
67     deviceId_ = -1;
68     targetDisplayId_ = -1;
69     targetWindowId_ = -1;
70     agentWindowId_ = -1;
71     bitwise_ = EVENT_FLAG_NONE;
72     markEnabled_ = true;
73 }
74 
ToString()75 std::string InputEvent::ToString()
76 {
77     std::string eventStr = "eventType:" + std::to_string(eventType_);
78     eventStr += ",actionTime:" + std::to_string(actionTime_);
79     eventStr += ",deviceId:" + std::to_string(deviceId_);
80     return eventStr;
81 }
82 
Create()83 std::shared_ptr<InputEvent> InputEvent::Create()
84 {
85     auto event = std::shared_ptr<InputEvent>(new (std::nothrow) InputEvent(InputEvent::EVENT_TYPE_BASE));
86     CHKPP(event);
87     return event;
88 }
89 
EventTypeToString(int32_t eventType)90 const char* InputEvent::EventTypeToString(int32_t eventType)
91 {
92     switch (eventType) {
93         case InputEvent::EVENT_TYPE_BASE: {
94             return "base";
95         }
96         case InputEvent::EVENT_TYPE_KEY: {
97             return "key";
98         }
99         case InputEvent::EVENT_TYPE_POINTER: {
100             return "pointer";
101         }
102         case InputEvent::EVENT_TYPE_AXIS: {
103             return "axis";
104         }
105         case InputEvent::EVENT_TYPE_FINGERPRINT: {
106             return "fingerprint";
107         }
108         default: {
109             MMI_HILOGW("Unknown EVENT_TYPE");
110             return "unknown";
111         }
112     }
113 }
114 
GetId() const115 int32_t InputEvent::GetId() const
116 {
117     return id_;
118 }
119 
SetId(int32_t id)120 void InputEvent::SetId(int32_t id)
121 {
122     id_ = id;
123 }
124 
UpdateId()125 void InputEvent::UpdateId()
126 {
127     id_ = g_nextEventId++;
128 }
129 
GetActionTime() const130 int64_t InputEvent::GetActionTime() const
131 {
132     return actionTime_;
133 }
134 
SetActionTime(int64_t actionTime)135 void InputEvent::SetActionTime(int64_t actionTime)
136 {
137     actionTime_ = actionTime;
138 }
139 
SetSensorInputTime(uint64_t sensorInputTime)140 void InputEvent::SetSensorInputTime(uint64_t sensorInputTime)
141 {
142     sensorInputTime_ = sensorInputTime;
143 }
144 
GetSensorInputTime()145 uint64_t InputEvent::GetSensorInputTime()
146 {
147     return sensorInputTime_;
148 }
149 
GetAction() const150 int32_t InputEvent::GetAction() const
151 {
152     return action_;
153 }
154 
SetAction(int32_t action)155 void InputEvent::SetAction(int32_t action)
156 {
157     action_ = action;
158 }
159 
GetActionStartTime() const160 int64_t InputEvent::GetActionStartTime() const
161 {
162     return actionStartTime_;
163 }
164 
SetActionStartTime(int64_t actionStartTime)165 void InputEvent::SetActionStartTime(int64_t actionStartTime)
166 {
167     actionStartTime_ = actionStartTime;
168 }
169 
GetDeviceId() const170 int32_t InputEvent::GetDeviceId() const
171 {
172     return deviceId_;
173 }
174 
SetDeviceId(int32_t deviceId)175 void InputEvent::SetDeviceId(int32_t deviceId)
176 {
177     deviceId_ = deviceId;
178 }
179 
GetTargetDisplayId() const180 int32_t InputEvent::GetTargetDisplayId() const
181 {
182     return targetDisplayId_;
183 }
184 
SetTargetDisplayId(int32_t displayId)185 void InputEvent::SetTargetDisplayId(int32_t displayId)
186 {
187     targetDisplayId_ = displayId;
188 }
189 
GetAgentWindowId() const190 int32_t InputEvent::GetAgentWindowId() const
191 {
192     return agentWindowId_;
193 }
194 
SetAgentWindowId(int32_t windowId)195 void InputEvent::SetAgentWindowId(int32_t windowId)
196 {
197     agentWindowId_ = windowId;
198 }
199 
GetTargetWindowId() const200 int32_t InputEvent::GetTargetWindowId() const
201 {
202     return targetWindowId_;
203 }
204 
SetTargetWindowId(int32_t windowId)205 void InputEvent::SetTargetWindowId(int32_t windowId)
206 {
207     targetWindowId_ = windowId;
208 }
209 
GetEventType() const210 int32_t InputEvent::GetEventType() const
211 {
212     return eventType_;
213 }
214 
GetFlag() const215 uint32_t InputEvent::GetFlag() const
216 {
217     return bitwise_;
218 }
219 
HasFlag(uint32_t flag)220 bool InputEvent::HasFlag(uint32_t flag)
221 {
222     return (bitwise_ & flag) != 0;
223 }
224 
AddFlag(uint32_t flag)225 void InputEvent::AddFlag(uint32_t flag)
226 {
227     bitwise_ |= flag;
228 }
229 
ClearFlag()230 void InputEvent::ClearFlag()
231 {
232     bitwise_ = EVENT_FLAG_NONE;
233 }
234 
ClearFlag(uint32_t flag)235 void InputEvent::ClearFlag(uint32_t flag)
236 {
237     bitwise_ &= (~flag);
238 }
239 
IsMarkEnabled() const240 bool InputEvent::IsMarkEnabled() const
241 {
242     return markEnabled_;
243 }
244 
245 
SetMarkEnabled(bool markEnabled)246 void InputEvent::SetMarkEnabled(bool markEnabled)
247 {
248     markEnabled_ = markEnabled;
249 }
250 
251 
SetProcessedCallback(std::function<void(int32_t, int64_t)> callback)252 void InputEvent::SetProcessedCallback(std::function<void(int32_t, int64_t)> callback)
253 {
254     processedCallback_ = callback;
255 }
256 
MarkProcessed()257 void InputEvent::MarkProcessed()
258 {
259     if (!processedCallback_) {
260         return;
261     }
262     if (!markEnabled_) {
263         MMI_HILOGD("Skip MarkProcessed eventId:%{public}d, eventType:%{public}d", id_, eventType_);
264         return;
265     }
266     auto func = processedCallback_;
267     processedCallback_ = std::function<void(int32_t, int64_t)>();
268     func(id_, actionTime_);
269 }
270 
SetExtraData(const std::shared_ptr<const uint8_t[]> data, uint32_t length)271 void InputEvent::SetExtraData(const std::shared_ptr<const uint8_t[]> data, uint32_t length)
272 {
273     if (data && (length > 0) && (length <= DATA_LENGTH_LIMIT)) {
274         extraData_ = data;
275         extraDataLength_ = length;
276     }
277 }
278 
GetExtraData(std::shared_ptr<const uint8_t[]> &data, uint32_t &length) const279 void InputEvent::GetExtraData(std::shared_ptr<const uint8_t[]> &data, uint32_t &length) const
280 {
281     if (extraData_ && extraDataLength_ != 0) {
282         data = extraData_;
283         length = extraDataLength_;
284     } else {
285         length = 0;
286     }
287 }
288 
WriteToParcel(Parcel &out) const289 bool InputEvent::WriteToParcel(Parcel &out) const
290 {
291     WRITEINT32(out, eventType_);
292     WRITEINT32(out, id_);
293     WRITEINT64(out, actionTime_);
294     WRITEUINT64(out, sensorInputTime_);
295     WRITEINT32(out, action_);
296     WRITEINT64(out, actionStartTime_);
297     WRITEINT32(out, deviceId_);
298     WRITEINT32(out, targetDisplayId_);
299     WRITEINT32(out, targetWindowId_);
300     WRITEINT32(out, agentWindowId_);
301     WRITEUINT32(out, bitwise_);
302     WRITEBOOL(out, markEnabled_);
303     if (extraData_ && extraDataLength_ != 0) {
304         WRITEUINT32(out, extraDataLength_);
305         WRITEBUFFER(out, (void *)extraData_.get(), extraDataLength_);
306     } else {
307         WRITEUINT32(out, 0);
308     }
309     return true;
310 }
311 
ReadFromParcel(Parcel &in)312 bool InputEvent::ReadFromParcel(Parcel &in)
313 {
314 #if defined(ANDROID_PLATFORM) || defined(IOS_PLATFORM)
315     return false;
316 #else
317     READINT32(in, eventType_);
318     READINT32(in, id_);
319     READINT64(in, actionTime_);
320     READUINT64(in, sensorInputTime_);
321     READINT32(in, action_);
322     READINT64(in, actionStartTime_);
323     READINT32(in, deviceId_);
324     READINT32(in, targetDisplayId_);
325     READINT32(in, targetWindowId_);
326     READINT32(in, agentWindowId_);
327     READUINT32(in, bitwise_);
328     READBOOL(in, markEnabled_);
329     READUINT32(in, extraDataLength_);
330 
331     if (extraDataLength_ == 0) {
332         return true;
333     }
334     if (extraDataLength_ > DATA_LENGTH_LIMIT) {
335         extraDataLength_ = 0;
336         return false;
337     }
338     const uint8_t *buffer = in.ReadBuffer(extraDataLength_);
339     if (buffer == nullptr) {
340         extraDataLength_ = 0;
341         return false;
342     }
343     std::shared_ptr<uint8_t[]> sp(new uint8_t[extraDataLength_], [](uint8_t* ptr) { delete[] ptr; });
344     if (sp == nullptr) {
345         extraDataLength_ = 0;
346         return false;
347     }
348     std::copy(buffer, buffer + extraDataLength_, sp.get());
349     extraData_ = sp;
350     return true;
351 #endif // defined(ANDROID_PLATFORM) || defined(IOS_PLATFORM)
352 }
353 
ActionToShortStr(int32_t action)354 std::string_view InputEvent::ActionToShortStr(int32_t action)
355 {
356     switch (action) {
357         case InputEvent::ACTION_CANCEL:
358             return "B:C:";
359         case InputEvent::ACTION_UNKNOWN:
360             return "B:UK:";
361         default:
362             return "B:?:";
363     }
364 }
365 
366 struct LogTraceKey {
367     int64_t Id;
368     int32_t action;
369     int32_t evtType;
370 };
371 
372 thread_local std::vector<LogTraceKey> g_traceIds;
373 thread_local std::unordered_map<int64_t, size_t> g_traceIdToIdx;
374 thread_local std::string g_traceStr;
375 
Action2Str(int32_t eventType, int32_t action)376 std::string_view Action2Str(int32_t eventType, int32_t action)
377 {
378     switch (eventType) {
379         case InputEvent::EVENT_TYPE_KEY: {
380             return KeyEvent::ActionToShortStr(action);
381         }
382         case InputEvent::EVENT_TYPE_POINTER:
383         case InputEvent::EVENT_TYPE_FINGERPRINT: {
384             return PointerEvent::ActionToShortStr(action);
385         }
386         case InputEvent::EVENT_TYPE_AXIS: {
387             return AxisEvent::ActionToShortStr(action);
388         }
389         case InputEvent::EVENT_TYPE_BASE: {
390             return InputEvent::ActionToShortStr(action);
391         }
392         default: {
393             return "?:?:";
394         }
395     }
396 }
397 
RefreshTraceStr()398 void RefreshTraceStr()
399 {
400     g_traceStr.clear();
401     for (auto item = g_traceIds.begin(); item < g_traceIds.end(); ++item) {
402         if (item->Id == -1) {
403             continue;
404         }
405         if (item != g_traceIds.begin()) {
406             g_traceStr += "/";
407         }
408         g_traceStr += Action2Str(item->evtType, item->action);
409         g_traceStr += std::to_string(item->Id);
410     }
411 }
412 
StartLogTraceId(int64_t traceId, int32_t eventType, int32_t action)413 void StartLogTraceId(int64_t traceId, int32_t eventType, int32_t action)
414 {
415     if (traceId == -1) {
416         return;
417     }
418     auto iter = g_traceIdToIdx.find(traceId);
419     if (iter == g_traceIdToIdx.end()) {
420         g_traceIds.push_back({traceId, action, eventType});
421         g_traceIdToIdx.emplace(traceId, g_traceIds.size() - 1);
422         std::string currentTraceStr(Action2Str(eventType, action));
423         currentTraceStr += std::to_string(traceId);
424         if (g_traceIds.size() == 1) {
425             g_traceStr = currentTraceStr;
426         } else {
427             g_traceStr += "/" + currentTraceStr;
428         }
429         return;
430     }
431     if (g_traceIds.size() <= iter->second) {
432         return;
433     }
434     LogTraceKey &old = g_traceIds.at(iter->second);
435     if (old.evtType != eventType || old.action != action) {
436         old.evtType = eventType;
437         old.action = action;
438         RefreshTraceStr();
439     }
440 };
441 
EndLogTraceId(int64_t id)442 void EndLogTraceId(int64_t id)
443 {
444     auto iter = g_traceIdToIdx.find(id);
445     if (iter == g_traceIdToIdx.end()) {
446         return;
447     }
448     size_t idx = iter->second;
449     g_traceIdToIdx.erase(iter);
450     size_t idCount = g_traceIds.size();
451     if (idCount <= idx) {
452         return;
453     }
454 
455     if (idCount == idx + 1) {
456         g_traceIds.pop_back();
457         while (!g_traceIds.empty() && g_traceIds.back().Id == -1) {
458             g_traceIds.pop_back();
459         }
460     } else {
461         // can't erase it, erase it will make the index of other elem changed.
462         LogTraceKey &toDelete = g_traceIds.at(idx);
463         toDelete.Id = -1;
464     }
465     RefreshTraceStr();
466 }
467 
FormatLogTrace()468 const char *FormatLogTrace()
469 {
470     return g_traceStr.c_str();
471 }
472 
473 
ResetLogTrace()474 void ResetLogTrace()
475 {
476     g_traceIds.clear();
477     g_traceIdToIdx.clear();
478     g_traceStr.clear();
479 }
480 
LogTracer(int64_t traceId, int32_t evtType, int32_t action)481 LogTracer::LogTracer(int64_t traceId, int32_t evtType, int32_t action)
482 {
483     traceId_ = traceId;
484     StartLogTraceId(traceId, evtType, action);
485 }
486 
~LogTracer()487 LogTracer::~LogTracer()
488 {
489     EndLogTraceId(traceId_);
490 }
491 
LogTracer()492 LogTracer::LogTracer()
493 {
494     traceId_ = -1;
495 }
496 
traceId_(other.traceId_)497 LogTracer::LogTracer(LogTracer &&other) noexcept: traceId_(other.traceId_)
498 {
499     other.traceId_ = -1;
500 }
501 
502 LogTracer &LogTracer::operator=(LogTracer &&other) noexcept
503 {
504     if (this != &other) {
505         traceId_ = other.traceId_;
506         other.traceId_ = -1;
507     }
508     return *this;
509 }
510 
511 int32_t OHOS::MMI::EventLogHelper::infoDictCount_ = 0;
512 int32_t OHOS::MMI::EventLogHelper::debugDictCount_ = 0;
513 
514 } // namespace MMI
515 } // namespace OHOS
516