2 * uterm - Linux User-Space Terminal
4 * Copyright (c) 2011 Ran Benita <ran234@gmail.com>
5 * Copyright (c) 2012 David Herrmann <dh.herrmann@googlemail.com>
7 * Permission is hereby granted, free of charge, to any person obtaining
8 * a copy of this software and associated documentation files
9 * (the "Software"), to deal in the Software without restriction, including
10 * without limitation the rights to use, copy, modify, merge, publish,
11 * distribute, sublicense, and/or sell copies of the Software, and to
12 * permit persons to whom the Software is furnished to do so, subject to
13 * the following conditions:
15 * The above copyright notice and this permission notice shall be included
16 * in all copies or substantial portions of the Software.
18 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
19 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
21 * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
22 * CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
23 * TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
24 * SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
29 #include <linux/input.h>
33 #include <xkbcommon/xkbcommon.h>
38 #include "uterm_input.h"
40 #define LOG_SUBSYSTEM "input_uxkb"
42 int uxkb_desc_init(struct uterm_input *input,
48 struct xkb_rule_names rmlvo = {
56 input->ctx = xkb_context_new(0);
58 log_error("cannot create XKB context");
62 input->keymap = xkb_keymap_new_from_names(input->ctx, &rmlvo, 0);
64 log_warn("failed to create keymap (%s, %s, %s), "
65 "reverting to default US keymap",
66 layout, variant, options);
72 input->keymap = xkb_keymap_new_from_names(input->ctx,
75 log_warn("failed to create XKB keymap");
81 log_debug("new keyboard description (%s, %s, %s)",
82 layout, variant, options);
86 xkb_context_unref(input->ctx);
90 void uxkb_desc_destroy(struct uterm_input *input)
92 xkb_keymap_unref(input->keymap);
93 xkb_context_unref(input->ctx);
96 static void timer_event(struct ev_timer *timer, uint64_t num, void *data)
98 struct uterm_input_dev *dev = data;
100 dev->repeat_event.handled = false;
101 shl_hook_call(dev->input->hook, dev->input, &dev->repeat_event);
104 int uxkb_dev_init(struct uterm_input_dev *dev)
108 ret = ev_eloop_new_timer(dev->input->eloop, &dev->repeat_timer, NULL,
113 dev->state = xkb_state_new(dev->input->keymap);
115 log_error("cannot create XKB state");
123 ev_eloop_rm_timer(dev->repeat_timer);
127 void uxkb_dev_destroy(struct uterm_input_dev *dev)
129 xkb_state_unref(dev->state);
130 ev_eloop_rm_timer(dev->repeat_timer);
133 #define EVDEV_KEYCODE_OFFSET 8
140 static inline int uxkb_dev_resize_event(struct uterm_input_dev *dev, size_t s)
144 if (s > dev->num_syms) {
145 tmp = realloc(dev->event.keysyms,
146 sizeof(uint32_t) * s);
148 log_warning("cannot reallocate keysym buffer");
151 dev->event.keysyms = tmp;
153 tmp = realloc(dev->event.codepoints,
154 sizeof(uint32_t) * s);
156 log_warning("cannot reallocate codepoints buffer");
159 dev->event.codepoints = tmp;
161 tmp = realloc(dev->repeat_event.keysyms,
162 sizeof(uint32_t) * s);
164 log_warning("cannot reallocate keysym buffer");
167 dev->repeat_event.keysyms = tmp;
169 tmp = realloc(dev->repeat_event.codepoints,
170 sizeof(uint32_t) * s);
172 log_warning("cannot reallocate codepoints buffer");
175 dev->repeat_event.codepoints = tmp;
183 static int uxkb_dev_fill_event(struct uterm_input_dev *dev,
184 struct uterm_input_event *ev,
187 const xkb_keysym_t *syms)
191 ret = uxkb_dev_resize_event(dev, num_syms);
196 ev->ascii = shl_get_ascii(dev->state, code, syms, num_syms);
197 ev->mods = shl_get_xkb_mods(dev->state);
198 ev->num_syms = num_syms;
199 memcpy(ev->keysyms, syms, sizeof(uint32_t) * num_syms);
201 for (i = 0; i < num_syms; ++i) {
202 ev->codepoints[i] = xkb_keysym_to_utf32(syms[i]);
203 if (!ev->codepoints[i])
204 ev->codepoints[i] = UTERM_INPUT_INVALID;
210 static void uxkb_dev_repeat(struct uterm_input_dev *dev, unsigned int state)
212 struct xkb_keymap *keymap = xkb_state_get_keymap(dev->state);
214 int num_keysyms, ret;
215 const uint32_t *keysyms;
216 struct itimerspec spec;
218 if (dev->repeating && dev->repeat_event.keycode == dev->event.keycode) {
219 if (state == KEY_RELEASED) {
220 dev->repeating = false;
221 ev_timer_update(dev->repeat_timer, NULL);
226 if (state == KEY_PRESSED &&
227 xkb_keymap_key_repeats(keymap, dev->event.keycode)) {
228 dev->repeat_event.keycode = dev->event.keycode;
229 dev->repeat_event.ascii = dev->event.ascii;
230 dev->repeat_event.mods = dev->event.mods;
231 dev->repeat_event.num_syms = dev->event.num_syms;
233 for (i = 0; i < dev->event.num_syms; ++i) {
234 dev->repeat_event.keysyms[i] = dev->event.keysyms[i];
235 dev->repeat_event.codepoints[i] =
236 dev->event.codepoints[i];
238 } else if (dev->repeating &&
239 !xkb_keymap_key_repeats(keymap, dev->event.keycode)) {
240 num_keysyms = xkb_state_key_get_syms(dev->state,
241 dev->repeat_event.keycode,
243 if (num_keysyms <= 0)
246 ret = uxkb_dev_fill_event(dev, &dev->repeat_event,
247 dev->repeat_event.keycode,
248 num_keysyms, keysyms);
257 dev->repeating = true;
258 spec.it_interval.tv_sec = 0;
259 spec.it_interval.tv_nsec = dev->input->repeat_rate * 1000000;
260 spec.it_value.tv_sec = 0;
261 spec.it_value.tv_nsec = dev->input->repeat_delay * 1000000;
262 ev_timer_update(dev->repeat_timer, &spec);
265 int uxkb_dev_process(struct uterm_input_dev *dev,
266 uint16_t key_state, uint16_t code)
268 struct xkb_state *state;
269 xkb_keycode_t keycode;
270 const xkb_keysym_t *keysyms;
271 int num_keysyms, ret;
273 if (key_state == KEY_REPEATED)
277 keycode = code + EVDEV_KEYCODE_OFFSET;
279 num_keysyms = xkb_state_key_get_syms(state, keycode, &keysyms);
281 if (key_state == KEY_PRESSED)
282 xkb_state_update_key(state, keycode, XKB_KEY_DOWN);
283 else if (key_state == KEY_RELEASED)
284 xkb_state_update_key(state, keycode, XKB_KEY_UP);
286 if (num_keysyms <= 0)
289 ret = uxkb_dev_fill_event(dev, &dev->event, keycode, num_keysyms,
294 uxkb_dev_repeat(dev, key_state);
296 if (key_state == KEY_RELEASED)
299 dev->event.handled = false;
300 shl_hook_call(dev->input->hook, dev->input, &dev->event);
306 * Call this when we regain control of the keyboard after losing it.
307 * We don't reset the locked group, this should survive a VT switch, etc. The
308 * locked modifiers are reset according to the keyboard LEDs.
310 void uxkb_dev_reset(struct uterm_input_dev *dev, const unsigned long *ledbits)
313 struct xkb_state *state;
314 static const struct {
318 { LED_NUML, XKB_LED_NAME_NUM },
319 { LED_CAPSL, XKB_LED_NAME_CAPS },
320 { LED_SCROLLL, XKB_LED_NAME_SCROLL },
323 /* TODO: Urghs, while the input device was closed we might have missed
324 * some events that affect internal state. As xkbcommon does not provide
325 * a way to reset the internal state, we simply recreate the state. This
326 * should have the same effect.
327 * It also has a bug that if the CTRL-Release event is skipped, then
328 * every further release will never perform a _real_ release. Kind of
329 * buggy so we should fix it upstream. */
330 state = xkb_state_new(dev->input->keymap);
332 log_warning("cannot recreate xkb-state");
335 xkb_state_unref(dev->state);
338 for (i = 0; i < sizeof(led_names) / sizeof(*led_names); i++) {
339 if (!input_bit_is_set(ledbits, led_names[i].led))
343 * TODO: Add support in xkbcommon for setting the led state,
344 * and updating the modifier state accordingly. E.g., something
346 * xkb_state_led_name_set_active(state, led_names[i].led);