| #pragma once | 
 |  | 
 | #include <fcntl.h> | 
 | #include <libevdev/libevdev.h> | 
 | #include <unistd.h> | 
 |  | 
 | #include <phosphor-logging/elog-errors.hpp> | 
 | #include <phosphor-logging/elog.hpp> | 
 | #include <xyz/openbmc_project/Common/error.hpp> | 
 |  | 
 | #include <memory> | 
 | #include <string> | 
 | #include <tuple> | 
 |  | 
 | namespace evdevpp | 
 | { | 
 | namespace evdev | 
 | { | 
 |  | 
 | using EvDevPtr = libevdev*; | 
 |  | 
 | namespace details | 
 | { | 
 |  | 
 | /** @brief unique_ptr functor to release an evdev reference. */ | 
 | struct EvDevDeleter | 
 | { | 
 |     void operator()(libevdev* ptr) const | 
 |     { | 
 |         deleter(ptr); | 
 |     } | 
 |  | 
 |     decltype(&libevdev_free) deleter = libevdev_free; | 
 | }; | 
 |  | 
 | /* @brief Alias evdev to a unique_ptr type for auto-release. */ | 
 | using EvDev = std::unique_ptr<libevdev, EvDevDeleter>; | 
 |  | 
 | } // namespace details | 
 |  | 
 | using namespace phosphor::logging; | 
 | /** @class EvDev | 
 |  *  @brief Provides C++ bindings to the libevdev C API. | 
 |  */ | 
 | class EvDev | 
 | { | 
 |   private: | 
 |     using InternalFailure = | 
 |         sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; | 
 |  | 
 |   public: | 
 |     /* Define all of the basic class operations: | 
 |      *     Not allowed: | 
 |      *         - Default constructor to avoid nullptrs. | 
 |      *         - Copy operations due to internal unique_ptr. | 
 |      *     Allowed: | 
 |      *         - Move operations. | 
 |      *         - Destructor. | 
 |      */ | 
 |     EvDev() = delete; | 
 |     EvDev(const EvDev&) = delete; | 
 |     EvDev& operator=(const EvDev&) = delete; | 
 |     EvDev(EvDev&&) = default; | 
 |     EvDev& operator=(EvDev&&) = default; | 
 |     ~EvDev() = default; | 
 |  | 
 |     /** @brief Conversion constructor from evdev. */ | 
 |     explicit EvDev(EvDevPtr ptr) : evdev(ptr) {} | 
 |  | 
 |     /** @brief Get the current event state. */ | 
 |     auto fetch(unsigned int type, unsigned int code) | 
 |     { | 
 |         int val; | 
 |         auto rc = libevdev_fetch_event_value(evdev.get(), type, code, &val); | 
 |         if (!rc) | 
 |         { | 
 |             log<level::ERR>("Error in call to libevdev_fetch_event_value", | 
 |                             entry("TYPE=%d", type), entry("CODE=%d", code)); | 
 |             elog<InternalFailure>(); | 
 |         } | 
 |  | 
 |         return val; | 
 |     } | 
 |  | 
 |     /** @brief Get the next event. */ | 
 |     auto next() | 
 |     { | 
 |         struct input_event ev; | 
 |         while (true) | 
 |         { | 
 |             auto rc = libevdev_next_event(evdev.get(), | 
 |                                           LIBEVDEV_READ_FLAG_NORMAL, &ev); | 
 |             if (rc < 0) | 
 |             { | 
 |                 log<level::ERR>("Error in call to libevdev_next_event", | 
 |                                 entry("RC=%d", rc)); | 
 |                 elog<InternalFailure>(); | 
 |             } | 
 |  | 
 |             if (ev.type == EV_SYN && ev.code == SYN_REPORT) | 
 |                 continue; | 
 |  | 
 |             break; | 
 |         } | 
 |         return std::make_tuple(ev.type, ev.code, ev.value); | 
 |     } | 
 |  | 
 |   private: | 
 |     EvDevPtr get() | 
 |     { | 
 |         return evdev.get(); | 
 |     } | 
 |  | 
 |     details::EvDev evdev; | 
 | }; | 
 |  | 
 | inline auto newFromFD(int fd) | 
 | { | 
 |     using InternalFailure = | 
 |         sdbusplus::xyz::openbmc_project::Common::Error::InternalFailure; | 
 |  | 
 |     EvDevPtr dev = nullptr; | 
 |     auto rc = libevdev_new_from_fd(fd, &dev); | 
 |  | 
 |     if (rc) | 
 |     { | 
 |         log<level::ERR>("Error in call to libevdev_new_from_fd", | 
 |                         entry("RC=%d", rc), entry("FD=%d", fd)); | 
 |         elog<InternalFailure>(); | 
 |     } | 
 |  | 
 |     return EvDev(dev); | 
 | } | 
 | } // namespace evdev | 
 | } // namespace evdevpp |