2 * Copyright © 2013 Red Hat, Inc.
4 * Permission to use, copy, modify, distribute, and sell this software and its
5 * documentation for any purpose is hereby granted without fee, provided that
6 * the above copyright notice appear in all copies and that both that copyright
7 * notice and this permission notice appear in supporting documentation, and
8 * that the name of the copyright holders not be used in advertising or
9 * publicity pertaining to distribution of the software without specific,
10 * written prior permission. The copyright holders make no representations
11 * about the suitability of this software for any purpose. It is provided "as
12 * is" without express or implied warranty.
14 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
15 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
16 * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
17 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
18 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
19 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
23 #ifndef LIBEVDEV_INT_H
24 #define LIBEVDEV_INT_H
32 #include "libevdev-util.h"
35 #define ABS_MT_MIN ABS_MT_SLOT
36 #define ABS_MT_MAX ABS_MT_TOOL_Y
37 #define ABS_MT_CNT (ABS_MT_MAX - ABS_MT_MIN + 1)
38 #define LIBEVDEV_EXPORT __attribute__((visibility("default")))
39 #define ALIAS(_to) __attribute__((alias(#_to)))
43 * default state: SYNC_NONE
45 * SYNC_NONE → SYN_DROPPED or forced sync → SYNC_NEEDED
46 * SYNC_NEEDED → libevdev_next_event(LIBEVDEV_READ_FLAG_SYNC) → SYNC_IN_PROGRESS
47 * SYNC_NEEDED → libevdev_next_event(LIBEVDEV_READ_FLAG_SYNC_NONE) → SYNC_NONE
48 * SYNC_IN_PROGRESS → libevdev_next_event(LIBEVDEV_READ_FLAG_SYNC_NONE) → SYNC_NONE
49 * SYNC_IN_PROGRESS → no sync events left → SYNC_NONE
58 struct mt_sync_state {
71 unsigned long bits[NLONGS(EV_CNT)];
72 unsigned long props[NLONGS(INPUT_PROP_CNT)];
73 unsigned long key_bits[NLONGS(KEY_CNT)];
74 unsigned long rel_bits[NLONGS(REL_CNT)];
75 unsigned long abs_bits[NLONGS(ABS_CNT)];
76 unsigned long led_bits[NLONGS(LED_CNT)];
77 unsigned long msc_bits[NLONGS(MSC_CNT)];
78 unsigned long sw_bits[NLONGS(SW_CNT)];
79 unsigned long rep_bits[NLONGS(REP_CNT)]; /* convenience, always 1 */
80 unsigned long ff_bits[NLONGS(FF_CNT)];
81 unsigned long snd_bits[NLONGS(SND_CNT)];
82 unsigned long key_values[NLONGS(KEY_CNT)];
83 unsigned long led_values[NLONGS(LED_CNT)];
84 unsigned long sw_values[NLONGS(SW_CNT)];
85 struct input_absinfo abs_info[ABS_CNT];
86 int *mt_slot_vals; /* [num_slots * ABS_MT_CNT] */
87 int num_slots; /**< valid slots in mt_slot_vals */
89 int rep_values[REP_CNT];
91 enum SyncState sync_state;
92 enum libevdev_grab_mode grabbed;
94 struct input_event *queue;
95 size_t queue_size; /**< size of queue in elements */
96 size_t queue_next; /**< next event index */
97 size_t queue_nsync; /**< number of sync events */
99 struct timeval last_event_time;
102 struct mt_sync_state *mt_state;
103 size_t mt_state_sz; /* in bytes */
104 unsigned long *slot_update;
105 size_t slot_update_sz; /* in bytes */
106 unsigned long *tracking_id_changes;
107 size_t tracking_id_changes_sz; /* in bytes */
112 enum libevdev_log_priority priority; /** minimum logging priority */
113 libevdev_log_func_t handler; /** handler function */
114 void *userdata; /** user-defined data pointer */
116 extern struct logdata log_data;
118 #define log_msg_cond(priority, ...) \
120 if (libevdev_get_log_priority() >= priority) \
121 log_msg(priority, log_data.userdata, __FILE__, __LINE__, __func__, __VA_ARGS__); \
124 #define log_error(...) log_msg_cond(LIBEVDEV_LOG_ERROR, __VA_ARGS__)
125 #define log_info(...) log_msg_cond(LIBEVDEV_LOG_INFO, __VA_ARGS__)
126 #define log_dbg(...) log_msg_cond(LIBEVDEV_LOG_DEBUG, __VA_ARGS__)
127 #define log_bug(...) log_msg_cond(LIBEVDEV_LOG_ERROR, "BUG: "__VA_ARGS__)
130 log_msg(enum libevdev_log_priority priority,
132 const char *file, int line, const char *func,
133 const char *format, ...) LIBEVDEV_ATTRIBUTE_PRINTF(6, 7);
136 * @return a pointer to the next element in the queue, or NULL if the queue
139 static inline struct input_event*
140 queue_push(struct libevdev *dev)
142 if (dev->queue_next >= dev->queue_size)
145 return &dev->queue[dev->queue_next++];
149 * Set ev to the last element in the queue, removing it from the queue.
151 * @return 0 on success, 1 if the queue is empty.
154 queue_pop(struct libevdev *dev, struct input_event *ev)
156 if (dev->queue_next == 0)
159 *ev = dev->queue[--dev->queue_next];
165 queue_peek(struct libevdev *dev, size_t idx, struct input_event *ev)
167 if (dev->queue_next == 0 || idx > dev->queue_next)
169 *ev = dev->queue[idx];
175 * Shift the first n elements into ev and return the number of elements
177 * ev must be large enough to store n elements.
179 * @param ev The buffer to copy into, or NULL
180 * @return The number of elements in ev.
183 queue_shift_multiple(struct libevdev *dev, size_t n, struct input_event *ev)
187 if (dev->queue_next == 0)
190 n = min(n, dev->queue_next);
193 for (i = 0; i < n; i++)
194 ev[i] = dev->queue[i];
197 for (i = 0; i < dev->queue_next - n; i++)
198 dev->queue[i] = dev->queue[n + i];
200 dev->queue_next -= n;
205 * Set ev to the first element in the queue, shifting everything else
208 * @return 0 on success, 1 if the queue is empty.
211 queue_shift(struct libevdev *dev, struct input_event *ev)
213 return queue_shift_multiple(dev, 1, ev) == 1 ? 0 : 1;
217 queue_alloc(struct libevdev *dev, size_t size)
222 dev->queue = calloc(size, sizeof(struct input_event));
226 dev->queue_size = size;
232 queue_free(struct libevdev *dev)
240 queue_num_elements(struct libevdev *dev)
242 return dev->queue_next;
246 queue_size(struct libevdev *dev)
248 return dev->queue_size;
252 queue_num_free_elements(struct libevdev *dev)
254 if (dev->queue_size == 0)
257 return dev->queue_size - dev->queue_next;
260 static inline struct input_event *
261 queue_next_element(struct libevdev *dev)
263 if (dev->queue_next == dev->queue_size)
266 return &dev->queue[dev->queue_next];
270 queue_set_num_elements(struct libevdev *dev, size_t nelem)
272 if (nelem > dev->queue_size)
275 dev->queue_next = nelem;
280 #define max_mask(uc, lc) \
282 *mask = dev->lc##_bits; \
283 max = libevdev_event_type_get_max(type); \
288 type_to_mask_const(const struct libevdev *dev, unsigned int type, const unsigned long **mask)
311 type_to_mask(struct libevdev *dev, unsigned int type, unsigned long **mask)