/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/ /*** This file is part of systemd. Copyright (C) 2014 David Herrmann <dh.herrmann@gmail.com> systemd is free software; you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. systemd 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with systemd; If not, see <http://www.gnu.org/licenses/>. ***/ #pragma once #include <inttypes.h> #include <libudev.h> #include <linux/input.h> #include <stdbool.h> #include <stdlib.h> #include <systemd/sd-bus.h> #include <systemd/sd-event.h> #include <xkbcommon/xkbcommon.h> #include "hashmap.h" #include "idev.h" #include "list.h" #include "util.h" typedef struct idev_link idev_link; typedef struct idev_device_vtable idev_device_vtable; typedef struct idev_element idev_element; typedef struct idev_element_vtable idev_element_vtable; /* * Evdev Elements */ bool idev_is_evdev(idev_element *e); idev_element *idev_find_evdev(idev_session *s, dev_t devnum); int idev_evdev_new(idev_element **out, idev_session *s, struct udev_device *ud); /* * Keyboard Devices */ bool idev_is_keyboard(idev_device *d); idev_device *idev_find_keyboard(idev_session *s, const char *name); int idev_keyboard_new(idev_device **out, idev_session *s, const char *name); /* * Element Links */ struct idev_link { /* element-to-device connection */ LIST_FIELDS(idev_link, links_by_element); idev_element *element; /* device-to-element connection */ LIST_FIELDS(idev_link, links_by_device); idev_device *device; }; /* * Devices */ struct idev_device_vtable { void (*free) (idev_device *d); void (*attach) (idev_device *d, idev_link *l); void (*detach) (idev_device *d, idev_link *l); int (*feed) (idev_device *d, idev_data *data); }; struct idev_device { const idev_device_vtable *vtable; idev_session *session; char *name; LIST_HEAD(idev_link, links); bool public : 1; bool enabled : 1; }; #define IDEV_DEVICE_INIT(_vtable, _session) ((idev_device){ \ .vtable = (_vtable), \ .session = (_session), \ }) idev_device *idev_find_device(idev_session *s, const char *name); int idev_device_add(idev_device *d, const char *name); idev_device *idev_device_free(idev_device *d); DEFINE_TRIVIAL_CLEANUP_FUNC(idev_device*, idev_device_free); int idev_device_feed(idev_device *d, idev_data *data); void idev_device_feedback(idev_device *d, idev_data *data); /* * Elements */ struct idev_element_vtable { void (*free) (idev_element *e); void (*enable) (idev_element *e); void (*disable) (idev_element *e); void (*open) (idev_element *e); void (*close) (idev_element *e); void (*resume) (idev_element *e, int fd); void (*pause) (idev_element *e, const char *mode); void (*feedback) (idev_element *e, idev_data *data); }; struct idev_element { const idev_element_vtable *vtable; idev_session *session; unsigned long n_open; char *name; LIST_HEAD(idev_link, links); bool enabled : 1; bool readable : 1; bool writable : 1; }; #define IDEV_ELEMENT_INIT(_vtable, _session) ((idev_element){ \ .vtable = (_vtable), \ .session = (_session), \ }) idev_element *idev_find_element(idev_session *s, const char *name); int idev_element_add(idev_element *e, const char *name); idev_element *idev_element_free(idev_element *e); DEFINE_TRIVIAL_CLEANUP_FUNC(idev_element*, idev_element_free); int idev_element_feed(idev_element *e, idev_data *data); void idev_element_feedback(idev_element *e, idev_data *data); /* * Sessions */ struct idev_session { idev_context *context; char *name; char *path; sd_bus_slot *slot_resume_device; sd_bus_slot *slot_pause_device; Hashmap *element_map; Hashmap *device_map; idev_event_fn event_fn; void *userdata; bool custom : 1; bool managed : 1; bool enabled : 1; }; idev_session *idev_find_session(idev_context *c, const char *name); int idev_session_raise_device_data(idev_session *s, idev_device *d, idev_data *data); /* * Contexts */ struct idev_context { unsigned long ref; sd_event *event; sd_bus *sysbus; Hashmap *session_map; Hashmap *data_map; };