1c29fa5a6Sopenharmony_ci/*
2c29fa5a6Sopenharmony_ci * Copyright (c) 2024 Huawei Device Co., Ltd.
3c29fa5a6Sopenharmony_ci * Licensed under the Apache License, Version 2.0 (the "License");
4c29fa5a6Sopenharmony_ci * you may not use this file except in compliance with the License.
5c29fa5a6Sopenharmony_ci * You may obtain a copy of the License at
6c29fa5a6Sopenharmony_ci *
7c29fa5a6Sopenharmony_ci * http://www.apache.org/licenses/LICENSE-2.0
8c29fa5a6Sopenharmony_ci *
9c29fa5a6Sopenharmony_ci * Unless required by applicable law or agreed to in writing, software
10c29fa5a6Sopenharmony_ci * distributed under the License is distributed on an "AS IS" BASIS,
11c29fa5a6Sopenharmony_ci * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12c29fa5a6Sopenharmony_ci * See the License for the specific language governing permissions and
13c29fa5a6Sopenharmony_ci * limitations under the License.
14c29fa5a6Sopenharmony_ci */
15c29fa5a6Sopenharmony_ci
16c29fa5a6Sopenharmony_ci#include "libinput_wrapper.h"
17c29fa5a6Sopenharmony_ci
18c29fa5a6Sopenharmony_ci#include <cinttypes>
19c29fa5a6Sopenharmony_ci#include <climits>
20c29fa5a6Sopenharmony_ci#include <iostream>
21c29fa5a6Sopenharmony_ci
22c29fa5a6Sopenharmony_ci#include <dirent.h>
23c29fa5a6Sopenharmony_ci#include <fcntl.h>
24c29fa5a6Sopenharmony_ci#include <sys/epoll.h>
25c29fa5a6Sopenharmony_ci#include <unistd.h>
26c29fa5a6Sopenharmony_ci
27c29fa5a6Sopenharmony_ci#include "define_multimodal.h"
28c29fa5a6Sopenharmony_ci#include "util.h"
29c29fa5a6Sopenharmony_ci
30c29fa5a6Sopenharmony_ci#undef MMI_LOG_DOMAIN
31c29fa5a6Sopenharmony_ci#define MMI_LOG_DOMAIN MMI_LOG_SERVER
32c29fa5a6Sopenharmony_ci#undef MMI_LOG_TAG
33c29fa5a6Sopenharmony_ci#define MMI_LOG_TAG "LibinputWrapper"
34c29fa5a6Sopenharmony_ci
35c29fa5a6Sopenharmony_cinamespace OHOS {
36c29fa5a6Sopenharmony_cinamespace MMI {
37c29fa5a6Sopenharmony_cinamespace {
38c29fa5a6Sopenharmony_ciconstexpr int32_t WAIT_TIME_FOR_INPUT { 10 };
39c29fa5a6Sopenharmony_ciconstexpr int32_t MAX_RETRY_COUNT { 5 };
40c29fa5a6Sopenharmony_ci} // namespace
41c29fa5a6Sopenharmony_ci
42c29fa5a6Sopenharmony_ciLibinputWrapper::~LibinputWrapper()
43c29fa5a6Sopenharmony_ci{
44c29fa5a6Sopenharmony_ci    if (input_ != nullptr) {
45c29fa5a6Sopenharmony_ci        libinput_unref(input_);
46c29fa5a6Sopenharmony_ci    }
47c29fa5a6Sopenharmony_ci}
48c29fa5a6Sopenharmony_ci
49c29fa5a6Sopenharmony_ciconstexpr static libinput_interface LIBINPUT_INTERFACE = {
50c29fa5a6Sopenharmony_ci    .open_restricted = [](const char *path, int32_t flags, void *user_data)->int32_t {
51c29fa5a6Sopenharmony_ci        if (path == nullptr) {
52c29fa5a6Sopenharmony_ci            MMI_HILOGWK("Input device path is nullptr");
53c29fa5a6Sopenharmony_ci            return RET_ERR;
54c29fa5a6Sopenharmony_ci        }
55c29fa5a6Sopenharmony_ci        char realPath[PATH_MAX] = {};
56c29fa5a6Sopenharmony_ci        if (realpath(path, realPath) == nullptr) {
57c29fa5a6Sopenharmony_ci            std::this_thread::sleep_for(std::chrono::milliseconds(WAIT_TIME_FOR_INPUT));
58c29fa5a6Sopenharmony_ci            MMI_HILOGWK("The error path is %{private}s", path);
59c29fa5a6Sopenharmony_ci            return RET_ERR;
60c29fa5a6Sopenharmony_ci        }
61c29fa5a6Sopenharmony_ci        int32_t fd;
62c29fa5a6Sopenharmony_ci        for (int32_t i = 0; i < MAX_RETRY_COUNT; i++) {
63c29fa5a6Sopenharmony_ci            fd = open(realPath, flags);
64c29fa5a6Sopenharmony_ci            if (fd >= 0) {
65c29fa5a6Sopenharmony_ci                break;
66c29fa5a6Sopenharmony_ci            }
67c29fa5a6Sopenharmony_ci            std::this_thread::sleep_for(std::chrono::milliseconds(WAIT_TIME_FOR_INPUT));
68c29fa5a6Sopenharmony_ci        }
69c29fa5a6Sopenharmony_ci        int32_t errNo = errno;
70c29fa5a6Sopenharmony_ci        MMI_HILOGWK("Libinput .open_restricted path:%{private}s,fd:%{public}d,errno:%{public}d", path, fd, errNo);
71c29fa5a6Sopenharmony_ci        return fd < 0 ? RET_ERR : fd;
72c29fa5a6Sopenharmony_ci    },
73c29fa5a6Sopenharmony_ci    .close_restricted = [](int32_t fd, void *user_data)
74c29fa5a6Sopenharmony_ci    {
75c29fa5a6Sopenharmony_ci        if (fd < 0) {
76c29fa5a6Sopenharmony_ci            return;
77c29fa5a6Sopenharmony_ci        }
78c29fa5a6Sopenharmony_ci        MMI_HILOGI("Libinput .close_restricted fd:%{public}d", fd);
79c29fa5a6Sopenharmony_ci        close(fd);
80c29fa5a6Sopenharmony_ci    },
81c29fa5a6Sopenharmony_ci};
82c29fa5a6Sopenharmony_ci
83c29fa5a6Sopenharmony_cibool LibinputWrapper::Init()
84c29fa5a6Sopenharmony_ci{
85c29fa5a6Sopenharmony_ci    input_ = libinput_path_create_context(&LIBINPUT_INTERFACE, nullptr);
86c29fa5a6Sopenharmony_ci    CHKPF(input_);
87c29fa5a6Sopenharmony_ci    fd_ = libinput_get_fd(input_);
88c29fa5a6Sopenharmony_ci    if (fd_ < 0) {
89c29fa5a6Sopenharmony_ci        libinput_unref(input_);
90c29fa5a6Sopenharmony_ci        fd_ = -1;
91c29fa5a6Sopenharmony_ci        return false;
92c29fa5a6Sopenharmony_ci    }
93c29fa5a6Sopenharmony_ci    return true;
94c29fa5a6Sopenharmony_ci}
95c29fa5a6Sopenharmony_ci
96c29fa5a6Sopenharmony_cistruct libinput_event* LibinputWrapper::Dispatch()
97c29fa5a6Sopenharmony_ci{
98c29fa5a6Sopenharmony_ci    CHKPP(input_);
99c29fa5a6Sopenharmony_ci    if (libinput_dispatch(input_) != 0) {
100c29fa5a6Sopenharmony_ci        std::cout << "Failed to dispatch input event" << std::endl;
101c29fa5a6Sopenharmony_ci        return nullptr;
102c29fa5a6Sopenharmony_ci    }
103c29fa5a6Sopenharmony_ci    return libinput_get_event(input_);
104c29fa5a6Sopenharmony_ci}
105c29fa5a6Sopenharmony_ci
106c29fa5a6Sopenharmony_civoid LibinputWrapper::DrainEvents()
107c29fa5a6Sopenharmony_ci{
108c29fa5a6Sopenharmony_ci    CHKPV(input_);
109c29fa5a6Sopenharmony_ci    struct libinput_event *event;
110c29fa5a6Sopenharmony_ci
111c29fa5a6Sopenharmony_ci    libinput_dispatch(input_);
112c29fa5a6Sopenharmony_ci    while ((event = libinput_get_event(input_))) {
113c29fa5a6Sopenharmony_ci        libinput_event_destroy(event);
114c29fa5a6Sopenharmony_ci        libinput_dispatch(input_);
115c29fa5a6Sopenharmony_ci    }
116c29fa5a6Sopenharmony_ci}
117c29fa5a6Sopenharmony_ci
118c29fa5a6Sopenharmony_cibool LibinputWrapper::AddPath(const std::string &path)
119c29fa5a6Sopenharmony_ci{
120c29fa5a6Sopenharmony_ci    CHKPF(input_);
121c29fa5a6Sopenharmony_ci    auto pos = devices_.find(path);
122c29fa5a6Sopenharmony_ci    if (pos != devices_.end()) {
123c29fa5a6Sopenharmony_ci        return true;
124c29fa5a6Sopenharmony_ci    }
125c29fa5a6Sopenharmony_ci    libinput_device* device = libinput_path_add_device(input_, path.c_str());
126c29fa5a6Sopenharmony_ci    CHKPF(device);
127c29fa5a6Sopenharmony_ci    devices_[std::move(path)] = libinput_device_ref(device);
128c29fa5a6Sopenharmony_ci    return true;
129c29fa5a6Sopenharmony_ci}
130c29fa5a6Sopenharmony_ci
131c29fa5a6Sopenharmony_civoid LibinputWrapper::RemovePath(const std::string &path)
132c29fa5a6Sopenharmony_ci{
133c29fa5a6Sopenharmony_ci    CHKPV(input_);
134c29fa5a6Sopenharmony_ci    auto pos = devices_.find(path);
135c29fa5a6Sopenharmony_ci    if (pos != devices_.end()) {
136c29fa5a6Sopenharmony_ci        libinput_path_remove_device(pos->second);
137c29fa5a6Sopenharmony_ci        libinput_device_unref(pos->second);
138c29fa5a6Sopenharmony_ci        devices_.erase(pos);
139c29fa5a6Sopenharmony_ci    }
140c29fa5a6Sopenharmony_ci}
141c29fa5a6Sopenharmony_ci} // namespace MMI
142c29fa5a6Sopenharmony_ci} // namespace OHOS
143