2020-08-02 22:22:19 +00:00
|
|
|
/*
|
|
|
|
KWin - the KDE window manager
|
|
|
|
This file is part of the KDE project.
|
2014-08-14 12:43:57 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-FileCopyrightText: 2014 Martin Gräßlin <mgraesslin@kde.org>
|
2014-08-14 12:43:57 +00:00
|
|
|
|
2020-08-02 22:22:19 +00:00
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
*/
|
2014-08-14 12:43:57 +00:00
|
|
|
#include "context.h"
|
|
|
|
#include "events.h"
|
2016-02-05 11:50:49 +00:00
|
|
|
#include "libinput_logging.h"
|
2021-02-09 18:18:36 +00:00
|
|
|
|
2021-01-30 13:22:07 +00:00
|
|
|
#include "main.h"
|
|
|
|
#include "platform.h"
|
|
|
|
#include "session.h"
|
2021-02-09 18:18:36 +00:00
|
|
|
#include "udev.h"
|
2014-08-14 12:43:57 +00:00
|
|
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
2019-07-09 19:19:26 +00:00
|
|
|
#include <cerrno>
|
|
|
|
|
2014-08-14 12:43:57 +00:00
|
|
|
namespace KWin
|
|
|
|
{
|
|
|
|
namespace LibInput
|
|
|
|
{
|
|
|
|
|
2016-02-05 11:50:49 +00:00
|
|
|
static void libinputLogHandler(libinput *libinput, libinput_log_priority priority, const char *format, va_list args)
|
|
|
|
{
|
|
|
|
Q_UNUSED(libinput)
|
|
|
|
char buf[1024];
|
|
|
|
if (std::vsnprintf(buf, 1023, format, args) <= 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
switch (priority) {
|
|
|
|
case LIBINPUT_LOG_PRIORITY_DEBUG:
|
|
|
|
qCDebug(KWIN_LIBINPUT) << "Libinput:" << buf;
|
|
|
|
break;
|
|
|
|
case LIBINPUT_LOG_PRIORITY_INFO:
|
|
|
|
qCInfo(KWIN_LIBINPUT) << "Libinput:" << buf;
|
|
|
|
break;
|
|
|
|
case LIBINPUT_LOG_PRIORITY_ERROR:
|
|
|
|
default:
|
|
|
|
qCCritical(KWIN_LIBINPUT) << "Libinput:" << buf;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-08-14 12:43:57 +00:00
|
|
|
Context::Context(const Udev &udev)
|
|
|
|
: m_libinput(libinput_udev_create_context(&Context::s_interface, this, udev))
|
2014-08-15 10:51:31 +00:00
|
|
|
, m_suspended(false)
|
2014-08-14 12:43:57 +00:00
|
|
|
{
|
2016-02-05 11:50:49 +00:00
|
|
|
libinput_log_set_priority(m_libinput, LIBINPUT_LOG_PRIORITY_DEBUG);
|
|
|
|
libinput_log_set_handler(m_libinput, &libinputLogHandler);
|
2014-08-14 12:43:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Context::~Context()
|
|
|
|
{
|
|
|
|
if (m_libinput) {
|
|
|
|
libinput_unref(m_libinput);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Context::assignSeat(const char *seat)
|
|
|
|
{
|
|
|
|
if (!isValid()) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return libinput_udev_assign_seat(m_libinput, seat) == 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int Context::fileDescriptor()
|
|
|
|
{
|
|
|
|
if (!isValid()) {
|
2016-06-02 14:23:23 +00:00
|
|
|
return -1;
|
2014-08-14 12:43:57 +00:00
|
|
|
}
|
|
|
|
return libinput_get_fd(m_libinput);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Context::dispatch()
|
|
|
|
{
|
|
|
|
libinput_dispatch(m_libinput);
|
|
|
|
}
|
|
|
|
|
|
|
|
const struct libinput_interface Context::s_interface = {
|
|
|
|
Context::openRestrictedCallback,
|
|
|
|
Context::closeRestrictedCallBack
|
|
|
|
};
|
|
|
|
|
|
|
|
int Context::openRestrictedCallback(const char *path, int flags, void *user_data)
|
|
|
|
{
|
|
|
|
return ((Context*)user_data)->openRestricted(path, flags);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Context::closeRestrictedCallBack(int fd, void *user_data)
|
|
|
|
{
|
|
|
|
((Context*)user_data)->closeRestricted(fd);
|
|
|
|
}
|
|
|
|
|
|
|
|
int Context::openRestricted(const char *path, int flags)
|
|
|
|
{
|
2021-01-30 13:22:07 +00:00
|
|
|
int fd = kwinApp()->platform()->session()->openRestricted(path);
|
2014-08-15 07:30:08 +00:00
|
|
|
if (fd < 0) {
|
|
|
|
// failed
|
|
|
|
return fd;
|
|
|
|
}
|
|
|
|
// adjust flags - based on Weston (logind-util.c)
|
|
|
|
int fl = fcntl(fd, F_GETFL);
|
|
|
|
auto errorHandling = [fd, this]() {
|
|
|
|
close(fd);
|
|
|
|
closeRestricted(fd);
|
|
|
|
};
|
|
|
|
if (fl < 0) {
|
|
|
|
errorHandling();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & O_NONBLOCK) {
|
|
|
|
fl |= O_NONBLOCK;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fcntl(fd, F_SETFL, fl) < 0) {
|
|
|
|
errorHandling();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
fl = fcntl(fd, F_GETFD);
|
|
|
|
if (fl < 0) {
|
|
|
|
errorHandling();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!(flags & O_CLOEXEC)) {
|
|
|
|
fl &= ~FD_CLOEXEC;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fcntl(fd, F_SETFD, fl) < 0) {
|
|
|
|
errorHandling();
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
return fd;
|
2014-08-14 12:43:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void Context::closeRestricted(int fd)
|
|
|
|
{
|
2021-01-30 13:22:07 +00:00
|
|
|
kwinApp()->platform()->session()->closeRestricted(fd);
|
2014-08-14 12:43:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Event *Context::event()
|
|
|
|
{
|
|
|
|
return Event::create(libinput_get_event(m_libinput));
|
|
|
|
}
|
|
|
|
|
2014-08-15 10:51:31 +00:00
|
|
|
void Context::suspend()
|
|
|
|
{
|
|
|
|
if (m_suspended) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
libinput_suspend(m_libinput);
|
|
|
|
m_suspended = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Context::resume()
|
|
|
|
{
|
|
|
|
if (!m_suspended) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
libinput_resume(m_libinput);
|
|
|
|
m_suspended = false;
|
|
|
|
}
|
|
|
|
|
2014-08-14 12:43:57 +00:00
|
|
|
}
|
|
|
|
}
|