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