mirror of
https://github.com/tuxedocomputers/tuxedo-touchpad-switch.git
synced 2025-01-18 11:41:10 +01:00
Merge GNOME and KDE logic and decide in programm what to run
This commit is contained in:
parent
d0bad515f7
commit
4118ab6962
11 changed files with 411 additions and 380 deletions
|
@ -5,19 +5,16 @@ project(tuxedo-touchpad-switch)
|
|||
find_package(PkgConfig REQUIRED)
|
||||
pkg_check_modules(deps REQUIRED IMPORTED_TARGET gio-2.0 udev)
|
||||
|
||||
add_executable(tuxedo-touchpad-switch tuxedo-touchpad-switch.cpp)
|
||||
add_executable(tuxedo-touchpad-switch tuxedo-touchpad-switch.cpp setup-gnome.cpp setup-kde.cpp touchpad-control.cpp)
|
||||
target_link_libraries(tuxedo-touchpad-switch udev PkgConfig::deps)
|
||||
|
||||
add_executable(tuxedo-touchpad-switch-kde tuxedo-touchpad-switch-kde.cpp)
|
||||
target_link_libraries(tuxedo-touchpad-switch-kde udev PkgConfig::deps)
|
||||
|
||||
install(TARGETS tuxedo-touchpad-switch tuxedo-touchpad-switch-kde DESTINATION bin/)
|
||||
install(TARGETS tuxedo-touchpad-switch DESTINATION bin/)
|
||||
install(FILES res/99-tuxedo-touchpad-switch.rules DESTINATION lib/udev/rules.d/)
|
||||
install(FILES res/tuxedo-touchpad-switch-lockfile DESTINATION /etc/ PERMISSIONS OWNER_READ GROUP_READ WORLD_READ) # absolute path on purpose: implemented as such in tuxedo-touchpad-switch.cpp
|
||||
install(FILES res/tuxedo-touchpad-switch-gdm.desktop DESTINATION /usr/share/gdm/greeter/autostart/) # absolute path on purpose: gdm has no config dir in /usr/local/
|
||||
install(FILES res/tuxedo-touchpad-switch.desktop res/tuxedo-touchpad-switch-kde.desktop DESTINATION /etc/xdg/autostart/) # absolute path on purpose: $XDG_CONFIG_DIRS does not include a folder under /usr/ by default https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html#variables
|
||||
install(FILES res/tuxedo-touchpad-switch.desktop DESTINATION /usr/share/gdm/greeter/autostart/) # absolute path on purpose: gdm has no config dir in /usr/local/
|
||||
install(FILES res/tuxedo-touchpad-switch.desktop DESTINATION /etc/xdg/autostart/) # absolute path on purpose: $XDG_CONFIG_DIRS does not include a folder under /usr/ by default https://specifications.freedesktop.org/basedir-spec/basedir-spec-latest.html#variables
|
||||
|
||||
SET(CPACK_GENERATOR DEB RPM)
|
||||
SET(CPACK_GENERATOR DEB)
|
||||
SET(CPACK_PACKAGE_CONTACT "Werner Sembach <tux@tuxedocomputers.com>")
|
||||
SET(CPACK_PACKAGE_VERSION 0.0.1)
|
||||
SET(CPACK_PACKAGE_DESCRIPTION_SUMMARY "Hardware toggle for Tongfang/Uniwill i2c touchpads. This will toggle the touchpad-disabled-led.")
|
||||
|
|
|
@ -1,4 +0,0 @@
|
|||
[Desktop Entry]
|
||||
Name=TUXEDO Touchpad Switch
|
||||
Exec=tuxedo-touchpad-switch
|
||||
Type=Application
|
|
@ -1,5 +0,0 @@
|
|||
[Desktop Entry]
|
||||
Name=TUXEDO Touchpad Switch
|
||||
Exec=tuxedo-touchpad-switch
|
||||
Type=Application
|
||||
OnlyShowIn=KDE;
|
|
@ -2,4 +2,3 @@
|
|||
Name=TUXEDO Touchpad Switch
|
||||
Exec=tuxedo-touchpad-switch
|
||||
Type=Application
|
||||
OnlyShowIn=GNOME;
|
||||
|
|
142
setup-gnome.cpp
Normal file
142
setup-gnome.cpp
Normal file
|
@ -0,0 +1,142 @@
|
|||
// Copyright (c) 2020 TUXEDO Computers GmbH <tux@tuxedocomputers.com>
|
||||
//
|
||||
// This file is part of TUXEDO Touchpad Switch.
|
||||
//
|
||||
// This file is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// TUXEDO Touchpad Switch is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with TUXEDO Touchpad Switch. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
#include "setup-gnome.h"
|
||||
|
||||
#include <iostream>
|
||||
|
||||
#include <sys/file.h>
|
||||
|
||||
#include <gio/gio.h>
|
||||
|
||||
#include "touchpad-control.h"
|
||||
|
||||
using std::cerr;
|
||||
using std::endl;
|
||||
|
||||
static int lockfile = -1;
|
||||
|
||||
static void send_events_handler(GSettings *settings, const char* key, __attribute__((unused)) gpointer user_data) {
|
||||
const gchar *send_events_string = g_settings_get_string(settings, key);
|
||||
if (!send_events_string) {
|
||||
cerr << "send_events_handler(...): g_settings_get_string(...) failed." << endl;
|
||||
return;
|
||||
}
|
||||
|
||||
int enabled = 0;
|
||||
if (send_events_string[0] == 'e') {
|
||||
enabled = 1;
|
||||
}
|
||||
|
||||
if (set_touchpad_state(enabled)) {
|
||||
cerr << "send_events_handler(...): set_touchpad_state(...) failed." << endl;
|
||||
}
|
||||
}
|
||||
|
||||
static void session_manager_properties_changed_handler(__attribute__((unused)) GDBusProxy *proxy, GVariant *changed_properties, __attribute__((unused)) GStrv invalidated_properties, gpointer user_data) {
|
||||
if (g_variant_is_of_type(changed_properties, G_VARIANT_TYPE_VARDICT)) {
|
||||
GVariantDict changed_properties_dict;
|
||||
gboolean sessionIsActive;
|
||||
|
||||
g_variant_dict_init (&changed_properties_dict, changed_properties);
|
||||
if (g_variant_dict_lookup (&changed_properties_dict, "SessionIsActive", "b", &sessionIsActive)) {
|
||||
if (sessionIsActive) {
|
||||
if (flock(lockfile, LOCK_EX)) {
|
||||
cerr << "properties_changed_handler(...): flock(...) failed." << endl;
|
||||
}
|
||||
send_events_handler((GSettings *)user_data, "send-events", NULL);
|
||||
}
|
||||
else {
|
||||
if (set_touchpad_state(1)) {
|
||||
cerr << "properties_changed_handler(...): set_touchpad_state(...) failed." << endl;
|
||||
}
|
||||
if (flock(lockfile, LOCK_UN)) {
|
||||
cerr << "properties_changed_handler(...): flock(...) failed." << endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
static void display_config_properties_changed_handler(__attribute__((unused)) GDBusProxy *proxy, GVariant *changed_properties, __attribute__((unused)) GStrv invalidated_properties, gpointer user_data) {
|
||||
if (g_variant_is_of_type(changed_properties, G_VARIANT_TYPE_VARDICT)) {
|
||||
GVariantDict changed_properties_dict;
|
||||
gint32 powerSaveMode;
|
||||
|
||||
g_variant_dict_init (&changed_properties_dict, changed_properties);
|
||||
if (g_variant_dict_lookup (&changed_properties_dict, "PowerSaveMode", "i", &powerSaveMode)) {
|
||||
if (powerSaveMode == 0) {
|
||||
send_events_handler((GSettings *)user_data, "send-events", NULL);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int setup_gnome(int lockfile_arg) {
|
||||
lockfile = lockfile_arg;
|
||||
|
||||
// get a new glib settings context to read the touchpad configuration of the current user
|
||||
GSettings *touchpad_settings = g_settings_new("org.gnome.desktop.peripherals.touchpad");
|
||||
if (!touchpad_settings) {
|
||||
cerr << "main(...): g_settings_new(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
// sync on config change
|
||||
if (g_signal_connect(touchpad_settings, "changed::send-events", G_CALLBACK(send_events_handler), NULL) < 1) {
|
||||
cerr << "main(...): g_signal_connect(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
// sync on xsession change
|
||||
GDBusProxy *session_manager_properties = g_dbus_proxy_new_for_bus_sync(G_BUS_TYPE_SESSION,
|
||||
G_DBUS_PROXY_FLAGS_NONE, NULL,
|
||||
"org.gnome.SessionManager",
|
||||
"/org/gnome/SessionManager",
|
||||
"org.gnome.SessionManager",
|
||||
NULL, NULL);
|
||||
if (session_manager_properties == NULL) {
|
||||
cerr << "main(...): g_dbus_proxy_new_for_bus_sync(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
if (g_signal_connect(session_manager_properties, "g-properties-changed", G_CALLBACK(session_manager_properties_changed_handler), touchpad_settings) < 1) {
|
||||
cerr << "main(...): g_signal_connect(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
// sync on wakeup
|
||||
GDBusProxy *display_config_properties = g_dbus_proxy_new_for_bus_sync(G_BUS_TYPE_SESSION,
|
||||
G_DBUS_PROXY_FLAGS_NONE, NULL,
|
||||
"org.gnome.Mutter.DisplayConfig",
|
||||
"/org/gnome/Mutter/DisplayConfig",
|
||||
"org.gnome.Mutter.DisplayConfig",
|
||||
NULL, NULL);
|
||||
if (display_config_properties == NULL) {
|
||||
cerr << "main(...): g_dbus_proxy_new_for_bus_sync(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
if (g_signal_connect(display_config_properties, "g-properties-changed", G_CALLBACK(display_config_properties_changed_handler), touchpad_settings) < 1) {
|
||||
cerr << "main(...): g_signal_connect(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
// sync on start
|
||||
// also ensures that "send-events" setting is accessed at least once, which is required for the GSettings singal handling to be correctly initialize
|
||||
send_events_handler(touchpad_settings, "send-events", NULL);
|
||||
|
||||
return EXIT_SUCCESS;
|
||||
}
|
20
setup-gnome.h
Normal file
20
setup-gnome.h
Normal file
|
@ -0,0 +1,20 @@
|
|||
// Copyright (c) 2020 TUXEDO Computers GmbH <tux@tuxedocomputers.com>
|
||||
//
|
||||
// This file is part of TUXEDO Touchpad Switch.
|
||||
//
|
||||
// This file is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// TUXEDO Touchpad Switch is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with TUXEDO Touchpad Switch. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
#pragma once
|
||||
|
||||
int setup_gnome(int lockfile);
|
|
@ -15,22 +15,16 @@
|
|||
// You should have received a copy of the GNU General Public License
|
||||
// along with TUXEDO Touchpad Switch. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
#include <linux/hidraw.h>
|
||||
#include <fcntl.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <sys/file.h>
|
||||
#include <signal.h>
|
||||
#include "setup-kde.h"
|
||||
|
||||
#include <libudev.h>
|
||||
#include <iostream>
|
||||
|
||||
#include <sys/file.h>
|
||||
|
||||
#include <gio/gio.h>
|
||||
|
||||
#include <iostream>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
#include "touchpad-control.h"
|
||||
|
||||
using std::cout;
|
||||
using std::cerr;
|
||||
using std::endl;
|
||||
|
||||
|
@ -38,107 +32,7 @@ int lockfile;
|
|||
gboolean isMousePluggedInPrev;
|
||||
gboolean isEnabledSave;
|
||||
|
||||
static int get_touchpad_hidraw_devices(std::vector<std::string> *devnodes) {
|
||||
int result = -EXIT_FAILURE;
|
||||
|
||||
struct udev *udev_context = udev_new();
|
||||
if (!udev_context) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_new(...) failed." << endl;
|
||||
return result;
|
||||
}
|
||||
|
||||
struct udev_enumerate *hidraw_devices = udev_enumerate_new(udev_context);
|
||||
if (!hidraw_devices) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_new(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
if (udev_enumerate_add_match_subsystem(hidraw_devices, "hidraw") < 0) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_add_match_subsystem(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
if (udev_enumerate_scan_devices(hidraw_devices) < 0) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_scan_devices(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
struct udev_list_entry *hidraw_devices_iterator = udev_enumerate_get_list_entry(hidraw_devices);
|
||||
if (!hidraw_devices_iterator) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_get_list_entry(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
struct udev_list_entry *hidraw_device_entry;
|
||||
udev_list_entry_foreach(hidraw_device_entry, hidraw_devices_iterator) {
|
||||
if (strstr(udev_list_entry_get_name(hidraw_device_entry), "i2c-UNIW0001:00")) {
|
||||
struct udev_device *hidraw_device = udev_device_new_from_syspath(udev_context, udev_list_entry_get_name(hidraw_device_entry));
|
||||
if (!hidraw_device) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_device_new_from_syspath(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
std::string devnode = udev_device_get_devnode(hidraw_device);
|
||||
devnodes->push_back(devnode);
|
||||
|
||||
udev_device_unref(hidraw_device);
|
||||
}
|
||||
}
|
||||
}
|
||||
result = devnodes->size();
|
||||
}
|
||||
}
|
||||
}
|
||||
udev_enumerate_unref(hidraw_devices);
|
||||
}
|
||||
udev_unref(udev_context);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
int set_touchpad_state(int enabled) {
|
||||
std::vector<std::string> devnodes;
|
||||
int touchpad_count = get_touchpad_hidraw_devices(&devnodes);
|
||||
if (touchpad_count < 0) {
|
||||
cerr << "send_events_handler(...): get_touchpad_hidraw_devices(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
if (touchpad_count == 0) {
|
||||
cout << "No compatible touchpads found." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
for (auto it = devnodes.begin(); it != devnodes.end(); ++it) {
|
||||
int hidraw = open((*it).c_str(), O_WRONLY|O_NONBLOCK);
|
||||
if (hidraw < 0) {
|
||||
cerr << "send_events_handler(...): open(\"" << *it << "\", O_WRONLY|O_NONBLOCK) failed." << endl;
|
||||
continue;
|
||||
}
|
||||
|
||||
// default is to enable touchpad, for this send "0x03" as feature report nr.7 (0x07) to the touchpad hid device
|
||||
char buffer[2] = {0x07, 0x00};
|
||||
// change 0x03 to 0x00 to disable touchpad when configuration string starts with [d]isable
|
||||
if (enabled) {
|
||||
buffer[1] = 0x03;
|
||||
}
|
||||
int result = ioctl(hidraw, HIDIOCSFEATURE(sizeof(buffer)/sizeof(buffer[0])), buffer);
|
||||
if (result < 0) {
|
||||
cerr << "send_events_handler(...): ioctl(...) on " << *it << " failed." << endl;
|
||||
close(hidraw);
|
||||
continue;
|
||||
}
|
||||
|
||||
close(hidraw);
|
||||
}
|
||||
|
||||
return EXIT_SUCCESS;
|
||||
}
|
||||
|
||||
void gracefull_exit(int signum) {
|
||||
int result = set_touchpad_state(1);
|
||||
if (flock(lockfile, LOCK_UN)) {
|
||||
cerr << "main(...): flock(...) failed." << endl;
|
||||
}
|
||||
close(lockfile);
|
||||
exit(result);
|
||||
}
|
||||
|
||||
void kded5_modules_touchpad_handler(GDBusProxy *proxy, __attribute__((unused)) char *sender_name, char *signal_name, GVariant *parameters, __attribute__((unused)) gpointer user_data) {
|
||||
static void kded5_modules_touchpad_handler(GDBusProxy *proxy, __attribute__((unused)) char *sender_name, char *signal_name, GVariant *parameters, __attribute__((unused)) gpointer user_data) {
|
||||
if (!strcmp("enabledChanged", signal_name) && g_variant_is_of_type(parameters, (const GVariantType *)"(b)") && g_variant_n_children(parameters)) {
|
||||
GVariant *enabledChanged = g_variant_get_child_value(parameters, 0);
|
||||
|
||||
|
@ -185,7 +79,7 @@ void kded5_modules_touchpad_handler(GDBusProxy *proxy, __attribute__((unused))
|
|||
}
|
||||
}
|
||||
|
||||
int kded5_modules_touchpad_init(GDBusProxy *proxy) {
|
||||
static int kded5_modules_touchpad_init(GDBusProxy *proxy) {
|
||||
GVariant *isEnabledParam = g_dbus_proxy_call_sync(proxy, "isEnabled", NULL, G_DBUS_CALL_FLAGS_NONE, G_MAXINT, NULL, NULL);
|
||||
if (isEnabledParam != NULL && g_variant_is_of_type(isEnabledParam, (const GVariantType *)"(b)") && g_variant_n_children(isEnabledParam)) {
|
||||
GVariant *isEnabled = g_variant_get_child_value(isEnabledParam, 0);
|
||||
|
@ -243,23 +137,9 @@ int kded5_modules_touchpad_init(GDBusProxy *proxy) {
|
|||
return EXIT_SUCCESS;
|
||||
}
|
||||
|
||||
int main() {
|
||||
lockfile = open("/etc/tuxedo-touchpad-switch-lockfile", O_RDONLY);
|
||||
if (lockfile < 0) {
|
||||
cerr << "main(...): open(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
}
|
||||
int setup_kde(int lockfile_arg) {
|
||||
lockfile = lockfile_arg;
|
||||
|
||||
//FIXME singal(...) deprecated -> man signal
|
||||
signal(SIGINT, gracefull_exit);
|
||||
signal(SIGTERM, gracefull_exit);
|
||||
|
||||
if (flock(lockfile, LOCK_EX)) {
|
||||
cerr << "main(...): flock(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
}
|
||||
|
||||
|
||||
// sync on config change, xsession change, and wakeup kde
|
||||
GDBusProxy *kded5_modules_touchpad = g_dbus_proxy_new_for_bus_sync(G_BUS_TYPE_SESSION,
|
||||
G_DBUS_PROXY_FLAGS_NONE, NULL,
|
||||
|
@ -269,30 +149,18 @@ int main() {
|
|||
NULL, NULL);
|
||||
if (kded5_modules_touchpad == NULL) {
|
||||
cerr << "main(...): g_dbus_proxy_new_for_bus_sync(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
if (g_signal_connect(kded5_modules_touchpad, "g-signal", G_CALLBACK(kded5_modules_touchpad_handler), NULL) < 1) {
|
||||
cerr << "main(...): g_signal_connect(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
|
||||
// sync on start
|
||||
if (kded5_modules_touchpad_init(kded5_modules_touchpad) == EXIT_FAILURE) {
|
||||
cerr << "main(...): kded5_modules_touchpad_init(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
|
||||
// start empty glib mainloop, required for glib signals to be catched
|
||||
GMainLoop *app = g_main_loop_new(NULL, TRUE);
|
||||
if (!app) {
|
||||
cerr << "main(...): g_main_loop_new(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
}
|
||||
|
||||
g_main_loop_run(app);
|
||||
// g_main_loop_run only returns on error
|
||||
cerr << "main(...): g_main_loop_run(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
return EXIT_SUCCESS;
|
||||
}
|
20
setup-kde.h
Normal file
20
setup-kde.h
Normal file
|
@ -0,0 +1,20 @@
|
|||
// Copyright (c) 2020 TUXEDO Computers GmbH <tux@tuxedocomputers.com>
|
||||
//
|
||||
// This file is part of TUXEDO Touchpad Switch.
|
||||
//
|
||||
// This file is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// TUXEDO Touchpad Switch is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with TUXEDO Touchpad Switch. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
#pragma once
|
||||
|
||||
int setup_kde(int lockfile);
|
135
touchpad-control.cpp
Normal file
135
touchpad-control.cpp
Normal file
|
@ -0,0 +1,135 @@
|
|||
// Copyright (c) 2020 TUXEDO Computers GmbH <tux@tuxedocomputers.com>
|
||||
//
|
||||
// This file is part of TUXEDO Touchpad Switch.
|
||||
//
|
||||
// This file is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// TUXEDO Touchpad Switch is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with TUXEDO Touchpad Switch. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
#include "touchpad-control.h"
|
||||
|
||||
#include <iostream>
|
||||
#include <vector>
|
||||
#include <string>
|
||||
|
||||
#include <cstring>
|
||||
|
||||
#include <fcntl.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <linux/hidraw.h>
|
||||
|
||||
#include <libudev.h>
|
||||
|
||||
using std::cout;
|
||||
using std::cerr;
|
||||
using std::endl;
|
||||
|
||||
// "std::vector<std::string> *devnodes" gets amended with found "/dev/hidraw*" device paths
|
||||
// returns -EXIT_FAILURE on error or the number of found "i2c-UNIW0001:00"-touchpads, starting with 0, meaning no touchpads found
|
||||
static int get_touchpad_hidraw_devices(std::vector<std::string> *devnodes) {
|
||||
int result = -EXIT_FAILURE;
|
||||
|
||||
struct udev *udev_context = udev_new();
|
||||
if (!udev_context) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_new(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
struct udev_enumerate *hidraw_devices = udev_enumerate_new(udev_context);
|
||||
if (!hidraw_devices) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_new(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
if (udev_enumerate_add_match_subsystem(hidraw_devices, "hidraw") < 0) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_add_match_subsystem(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
if (udev_enumerate_scan_devices(hidraw_devices) < 0) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_scan_devices(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
struct udev_list_entry *hidraw_devices_iterator = udev_enumerate_get_list_entry(hidraw_devices);
|
||||
if (!hidraw_devices_iterator) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_get_list_entry(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
struct udev_list_entry *hidraw_device_entry;
|
||||
udev_list_entry_foreach(hidraw_device_entry, hidraw_devices_iterator) {
|
||||
if (strstr(udev_list_entry_get_name(hidraw_device_entry), "i2c-UNIW0001:00")) {
|
||||
struct udev_device *hidraw_device = udev_device_new_from_syspath(udev_context, udev_list_entry_get_name(hidraw_device_entry));
|
||||
if (!hidraw_device) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_device_new_from_syspath(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
std::string devnode = udev_device_get_devnode(hidraw_device);
|
||||
devnodes->push_back(devnode);
|
||||
|
||||
udev_device_unref(hidraw_device);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
result = devnodes->size();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
udev_enumerate_unref(hidraw_devices);
|
||||
}
|
||||
|
||||
udev_unref(udev_context);
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
int set_touchpad_state(int enabled) {
|
||||
std::vector<std::string> devnodes;
|
||||
int touchpad_count = get_touchpad_hidraw_devices(&devnodes);
|
||||
if (touchpad_count < 0) {
|
||||
cerr << "send_events_handler(...): get_touchpad_hidraw_devices(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
if (touchpad_count == 0) {
|
||||
cout << "No compatible touchpads found." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
int result = EXIT_SUCCESS;
|
||||
|
||||
for (auto it = devnodes.begin(); it != devnodes.end(); ++it) {
|
||||
int hidraw = open((*it).c_str(), O_WRONLY|O_NONBLOCK);
|
||||
if (hidraw < 0) {
|
||||
cerr << "send_events_handler(...): open(\"" << *it << "\", O_WRONLY|O_NONBLOCK) failed." << endl;
|
||||
result = EXIT_FAILURE;
|
||||
}
|
||||
else {
|
||||
// to enable touchpad send "0x03" as feature report nr.7 (0x07) to the touchpad hid device
|
||||
// to disable it send "0x030"
|
||||
char buffer[2] = {0x07, 0x00};
|
||||
if (enabled) {
|
||||
buffer[1] = 0x03;
|
||||
}
|
||||
|
||||
int result = ioctl(hidraw, HIDIOCSFEATURE(sizeof(buffer)/sizeof(buffer[0])), buffer);
|
||||
if (result < 0) {
|
||||
cerr << "send_events_handler(...): ioctl(...) on " << *it << " failed." << endl;
|
||||
result = EXIT_FAILURE;
|
||||
}
|
||||
// else {}
|
||||
|
||||
close(hidraw);
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
22
touchpad-control.h
Normal file
22
touchpad-control.h
Normal file
|
@ -0,0 +1,22 @@
|
|||
// Copyright (c) 2020 TUXEDO Computers GmbH <tux@tuxedocomputers.com>
|
||||
//
|
||||
// This file is part of TUXEDO Touchpad Switch.
|
||||
//
|
||||
// This file is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU General Public License as published by
|
||||
// the Free Software Foundation, either version 3 of the License, or
|
||||
// (at your option) any later version.
|
||||
//
|
||||
// TUXEDO Touchpad Switch is distributed in the hope that it will be useful,
|
||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
// GNU General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU General Public License
|
||||
// along with TUXEDO Touchpad Switch. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
#pragma once
|
||||
|
||||
// "int enable" set to 0 disables the touchpad, any other value enables it
|
||||
// returns EXIT_SUCCESS or EXIT_FAILURE accordingly, on fail the activate/deactivate state of found touchpads is undefined
|
||||
int set_touchpad_state(int enabled);
|
|
@ -15,262 +15,99 @@
|
|||
// You should have received a copy of the GNU General Public License
|
||||
// along with TUXEDO Touchpad Switch. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
#include <linux/hidraw.h>
|
||||
#include <iostream>
|
||||
|
||||
#include <cstdlib>
|
||||
#include <csignal>
|
||||
|
||||
#include <fcntl.h>
|
||||
#include <unistd.h>
|
||||
#include <sys/ioctl.h>
|
||||
#include <sys/file.h>
|
||||
#include <signal.h>
|
||||
|
||||
#include <libudev.h>
|
||||
|
||||
#include <gio/gio.h>
|
||||
|
||||
#include <iostream>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
#include "touchpad-control.h"
|
||||
#include "setup-gnome.h"
|
||||
#include "setup-kde.h"
|
||||
|
||||
using std::cout;
|
||||
using std::cerr;
|
||||
using std::endl;
|
||||
|
||||
int lockfile;
|
||||
static int lockfile = -1;
|
||||
|
||||
static int get_touchpad_hidraw_devices(std::vector<std::string> *devnodes) {
|
||||
int result = -EXIT_FAILURE;
|
||||
|
||||
struct udev *udev_context = udev_new();
|
||||
if (!udev_context) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_new(...) failed." << endl;
|
||||
return result;
|
||||
}
|
||||
|
||||
struct udev_enumerate *hidraw_devices = udev_enumerate_new(udev_context);
|
||||
if (!hidraw_devices) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_new(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
if (udev_enumerate_add_match_subsystem(hidraw_devices, "hidraw") < 0) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_add_match_subsystem(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
if (udev_enumerate_scan_devices(hidraw_devices) < 0) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_scan_devices(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
struct udev_list_entry *hidraw_devices_iterator = udev_enumerate_get_list_entry(hidraw_devices);
|
||||
if (!hidraw_devices_iterator) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_enumerate_get_list_entry(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
struct udev_list_entry *hidraw_device_entry;
|
||||
udev_list_entry_foreach(hidraw_device_entry, hidraw_devices_iterator) {
|
||||
if (strstr(udev_list_entry_get_name(hidraw_device_entry), "i2c-UNIW0001:00")) {
|
||||
struct udev_device *hidraw_device = udev_device_new_from_syspath(udev_context, udev_list_entry_get_name(hidraw_device_entry));
|
||||
if (!hidraw_device) {
|
||||
cerr << "get_touchpad_hidraw_devices(...): udev_device_new_from_syspath(...) failed." << endl;
|
||||
}
|
||||
else {
|
||||
std::string devnode = udev_device_get_devnode(hidraw_device);
|
||||
devnodes->push_back(devnode);
|
||||
|
||||
udev_device_unref(hidraw_device);
|
||||
}
|
||||
}
|
||||
}
|
||||
result = devnodes->size();
|
||||
}
|
||||
}
|
||||
}
|
||||
udev_enumerate_unref(hidraw_devices);
|
||||
}
|
||||
udev_unref(udev_context);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
int set_touchpad_state(int enabled) {
|
||||
std::vector<std::string> devnodes;
|
||||
int touchpad_count = get_touchpad_hidraw_devices(&devnodes);
|
||||
if (touchpad_count < 0) {
|
||||
cerr << "send_events_handler(...): get_touchpad_hidraw_devices(...) failed." << endl;
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
if (touchpad_count == 0) {
|
||||
cout << "No compatible touchpads found." << endl;
|
||||
return EXIT_FAILURE;
|
||||
static void gracefull_exit(int signum = 0) {
|
||||
int result = EXIT_SUCCESS;
|
||||
if (signum < 0) {
|
||||
result = EXIT_FAILURE;
|
||||
}
|
||||
|
||||
for (auto it = devnodes.begin(); it != devnodes.end(); ++it) {
|
||||
int hidraw = open((*it).c_str(), O_WRONLY|O_NONBLOCK);
|
||||
if (hidraw < 0) {
|
||||
cerr << "send_events_handler(...): open(\"" << *it << "\", O_WRONLY|O_NONBLOCK) failed." << endl;
|
||||
continue;
|
||||
}
|
||||
|
||||
// default is to enable touchpad, for this send "0x03" as feature report nr.7 (0x07) to the touchpad hid device
|
||||
char buffer[2] = {0x07, 0x00};
|
||||
// change 0x03 to 0x00 to disable touchpad when configuration string starts with [d]isable
|
||||
if (enabled) {
|
||||
buffer[1] = 0x03;
|
||||
}
|
||||
int result = ioctl(hidraw, HIDIOCSFEATURE(sizeof(buffer)/sizeof(buffer[0])), buffer);
|
||||
if (result < 0) {
|
||||
cerr << "send_events_handler(...): ioctl(...) on " << *it << " failed." << endl;
|
||||
close(hidraw);
|
||||
continue;
|
||||
}
|
||||
|
||||
close(hidraw);
|
||||
if (set_touchpad_state(1) != EXIT_SUCCESS) {
|
||||
cerr << "gracefull_exit(...): set_touchpad_state(...) failed." << endl;
|
||||
result = EXIT_FAILURE;
|
||||
}
|
||||
|
||||
return EXIT_SUCCESS;
|
||||
}
|
||||
|
||||
void gracefull_exit(int signum) {
|
||||
int result = set_touchpad_state(1);
|
||||
if (flock(lockfile, LOCK_UN)) {
|
||||
cerr << "main(...): flock(...) failed." << endl;
|
||||
if (lockfile >= 0) {
|
||||
if (flock(lockfile, LOCK_UN)) {
|
||||
cerr << "gracefull_exit(...): flock(...) failed." << endl;
|
||||
result = EXIT_FAILURE;
|
||||
}
|
||||
if (close(lockfile)) {
|
||||
cerr << "gracefull_exit(...): close(...) failed." << endl;
|
||||
result = EXIT_FAILURE;
|
||||
}
|
||||
}
|
||||
close(lockfile);
|
||||
|
||||
exit(result);
|
||||
}
|
||||
|
||||
void send_events_handler(GSettings *settings, const char* key, __attribute__((unused)) gpointer user_data) {
|
||||
const gchar *send_events_string = g_settings_get_string(settings, key);
|
||||
if (!send_events_string) {
|
||||
cerr << "send_events_handler(...): g_settings_get_string(...) failed." << endl;
|
||||
return;
|
||||
}
|
||||
|
||||
int enabled = 0;
|
||||
if (send_events_string[0] == 'e') {
|
||||
enabled = 1;
|
||||
}
|
||||
|
||||
if (set_touchpad_state(enabled)) {
|
||||
cerr << "send_events_handler(...): set_touchpad_state(...) failed." << endl;
|
||||
}
|
||||
}
|
||||
|
||||
void session_manager_properties_changed_handler(__attribute__((unused)) GDBusProxy *proxy, GVariant *changed_properties, __attribute__((unused)) GStrv invalidated_properties, gpointer user_data) {
|
||||
if (g_variant_is_of_type(changed_properties, G_VARIANT_TYPE_VARDICT)) {
|
||||
GVariantDict changed_properties_dict;
|
||||
gboolean sessionIsActive;
|
||||
|
||||
g_variant_dict_init (&changed_properties_dict, changed_properties);
|
||||
if (g_variant_dict_lookup (&changed_properties_dict, "SessionIsActive", "b", &sessionIsActive)) {
|
||||
if (sessionIsActive) {
|
||||
if (flock(lockfile, LOCK_EX)) {
|
||||
cerr << "properties_changed_handler(...): flock(...) failed." << endl;
|
||||
}
|
||||
send_events_handler((GSettings *)user_data, "send-events", NULL);
|
||||
}
|
||||
else {
|
||||
if (set_touchpad_state(1)) {
|
||||
cerr << "properties_changed_handler(...): set_touchpad_state(...) failed." << endl;
|
||||
}
|
||||
if (flock(lockfile, LOCK_UN)) {
|
||||
cerr << "properties_changed_handler(...): flock(...) failed." << endl;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void display_config_properties_changed_handler(__attribute__((unused)) GDBusProxy *proxy, GVariant *changed_properties, __attribute__((unused)) GStrv invalidated_properties, gpointer user_data) {
|
||||
if (g_variant_is_of_type(changed_properties, G_VARIANT_TYPE_VARDICT)) {
|
||||
GVariantDict changed_properties_dict;
|
||||
gint32 powerSaveMode;
|
||||
|
||||
g_variant_dict_init (&changed_properties_dict, changed_properties);
|
||||
if (g_variant_dict_lookup (&changed_properties_dict, "PowerSaveMode", "i", &powerSaveMode)) {
|
||||
cout << powerSaveMode << endl;
|
||||
if (powerSaveMode == 0) {
|
||||
send_events_handler((GSettings *)user_data, "send-events", NULL);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
int main() {
|
||||
struct sigaction sigaction_gracefull_exit;
|
||||
sigaction_gracefull_exit.sa_handler = gracefull_exit;
|
||||
sigemptyset(&sigaction_gracefull_exit.sa_mask);
|
||||
sigaction_gracefull_exit.sa_flags = 0;
|
||||
|
||||
if (sigaction(SIGINT, &sigaction_gracefull_exit, nullptr)) {
|
||||
cerr << "main(...): sigaction(...) failed." << endl;
|
||||
gracefull_exit(-EXIT_FAILURE);
|
||||
}
|
||||
if (sigaction(SIGTERM, &sigaction_gracefull_exit, nullptr)) {
|
||||
cerr << "main(...): sigaction(...) failed." << endl;
|
||||
gracefull_exit(-EXIT_FAILURE);
|
||||
}
|
||||
|
||||
lockfile = open("/etc/tuxedo-touchpad-switch-lockfile", O_RDONLY);
|
||||
if (lockfile < 0) {
|
||||
if (lockfile == -1) {
|
||||
cerr << "main(...): open(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
gracefull_exit(-EXIT_FAILURE);
|
||||
}
|
||||
|
||||
//FIXME singal(...) deprecated -> man signal
|
||||
signal(SIGINT, gracefull_exit);
|
||||
signal(SIGTERM, gracefull_exit);
|
||||
|
||||
if (flock(lockfile, LOCK_EX)) {
|
||||
if (flock(lockfile, LOCK_EX) == -1) {
|
||||
cerr << "main(...): flock(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
gracefull_exit(-EXIT_FAILURE);
|
||||
}
|
||||
|
||||
// get a new glib settings context to read the touchpad configuration of the current user
|
||||
GSettings *touchpad_settings = g_settings_new("org.gnome.desktop.peripherals.touchpad");
|
||||
if (!touchpad_settings) {
|
||||
cerr << "main(...): g_settings_new(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
char *xdg_current_desktop = getenv("XDG_CURRENT_DESKTOP");
|
||||
if (strstr(xdg_current_desktop, "GNOME")) {
|
||||
setup_gnome(lockfile);
|
||||
}
|
||||
|
||||
|
||||
// sync on config change
|
||||
if (g_signal_connect(touchpad_settings, "changed::send-events", G_CALLBACK(send_events_handler), NULL) < 1) {
|
||||
cerr << "main(...): g_signal_connect(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
else if (strstr(xdg_current_desktop, "KDE")) {
|
||||
setup_kde(lockfile);
|
||||
}
|
||||
|
||||
|
||||
// sync on xsession change
|
||||
GDBusProxy *session_manager_properties = g_dbus_proxy_new_for_bus_sync(G_BUS_TYPE_SESSION,
|
||||
G_DBUS_PROXY_FLAGS_NONE, NULL,
|
||||
"org.gnome.SessionManager",
|
||||
"/org/gnome/SessionManager",
|
||||
"org.gnome.SessionManager",
|
||||
NULL, NULL);
|
||||
if (session_manager_properties == NULL) {
|
||||
cerr << "main(...): g_dbus_proxy_new_for_bus_sync(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
else {
|
||||
cout << "Your desktop environment is not supported." << endl;
|
||||
gracefull_exit(SIGTERM);
|
||||
}
|
||||
if (g_signal_connect(session_manager_properties, "g-properties-changed", G_CALLBACK(session_manager_properties_changed_handler), touchpad_settings) < 1) {
|
||||
cerr << "main(...): g_signal_connect(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
}
|
||||
|
||||
|
||||
// sync on wakeup
|
||||
GDBusProxy *display_config_properties = g_dbus_proxy_new_for_bus_sync(G_BUS_TYPE_SESSION,
|
||||
G_DBUS_PROXY_FLAGS_NONE, NULL,
|
||||
"org.gnome.Mutter.DisplayConfig",
|
||||
"/org/gnome/Mutter/DisplayConfig",
|
||||
"org.gnome.Mutter.DisplayConfig",
|
||||
NULL, NULL);
|
||||
if (display_config_properties == NULL) {
|
||||
cerr << "main(...): g_dbus_proxy_new_for_bus_sync(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
}
|
||||
if (g_signal_connect(display_config_properties, "g-properties-changed", G_CALLBACK(display_config_properties_changed_handler), touchpad_settings) < 1) {
|
||||
cerr << "main(...): g_signal_connect(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
}
|
||||
|
||||
|
||||
// sync on start
|
||||
send_events_handler(touchpad_settings, "send-events", NULL);
|
||||
|
||||
|
||||
// start empty glib mainloop, required for glib signals to be catched
|
||||
GMainLoop *app = g_main_loop_new(NULL, TRUE);
|
||||
if (!app) {
|
||||
cerr << "main(...): g_main_loop_new(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
gracefull_exit(-EXIT_FAILURE);
|
||||
}
|
||||
|
||||
g_main_loop_run(app);
|
||||
// g_main_loop_run only returns on error
|
||||
cerr << "main(...): g_main_loop_run(...) failed." << endl;
|
||||
gracefull_exit(0);
|
||||
gracefull_exit(-EXIT_FAILURE);
|
||||
}
|
||||
|
|
Loading…
Reference in a new issue