keymap: Keep keymap fd management in one place
At the same time, reduce the distance between this and the Xwayland handling branch.
This commit is contained in:
		@ -28,33 +28,24 @@
 | 
				
			|||||||
#include <sys/random.h> // TODO: this is Linux-specific
 | 
					#include <sys/random.h> // TODO: this is Linux-specific
 | 
				
			||||||
#include <xkbcommon/xkbcommon.h>
 | 
					#include <xkbcommon/xkbcommon.h>
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					 | 
				
			||||||
#include "eek-keyboard.h"
 | 
					#include "eek-keyboard.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void level_keyboard_free(LevelKeyboard *self) {
 | 
					
 | 
				
			||||||
    xkb_keymap_unref(self->keymap);
 | 
					static void eek_key_map_deinit(struct keymap *self) {
 | 
				
			||||||
    close(self->keymap_fd);
 | 
					    if (self->fd < 0) {
 | 
				
			||||||
    squeek_layout_free(self->layout);
 | 
					        g_error("Deinit called multiple times on KeyMap");
 | 
				
			||||||
    g_free(self);
 | 
					    } else {
 | 
				
			||||||
 | 
					        close(self->fd);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    self->fd = -1;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
LevelKeyboard*
 | 
					static struct keymap eek_key_map_from_str(const char *keymap_str) {
 | 
				
			||||||
level_keyboard_new (struct squeek_layout *layout)
 | 
					 | 
				
			||||||
{
 | 
					 | 
				
			||||||
    LevelKeyboard *keyboard = g_new0(LevelKeyboard, 1);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (!keyboard) {
 | 
					 | 
				
			||||||
        g_error("Failed to create a keyboard");
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    keyboard->layout = layout;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    struct xkb_context *context = xkb_context_new(XKB_CONTEXT_NO_FLAGS);
 | 
					    struct xkb_context *context = xkb_context_new(XKB_CONTEXT_NO_FLAGS);
 | 
				
			||||||
    if (!context) {
 | 
					    if (!context) {
 | 
				
			||||||
        g_error("No context created");
 | 
					        g_error("No context created");
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    const gchar *keymap_str = squeek_layout_get_keymap(keyboard->layout);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    struct xkb_keymap *keymap = xkb_keymap_new_from_string(context, keymap_str,
 | 
					    struct xkb_keymap *keymap = xkb_keymap_new_from_string(context, keymap_str,
 | 
				
			||||||
        XKB_KEYMAP_FORMAT_TEXT_V1, XKB_KEYMAP_COMPILE_NO_FLAGS);
 | 
					        XKB_KEYMAP_FORMAT_TEXT_V1, XKB_KEYMAP_COMPILE_NO_FLAGS);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -62,10 +53,9 @@ level_keyboard_new (struct squeek_layout *layout)
 | 
				
			|||||||
        g_error("Bad keymap:\n%s", keymap_str);
 | 
					        g_error("Bad keymap:\n%s", keymap_str);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    xkb_context_unref(context);
 | 
					    xkb_context_unref(context);
 | 
				
			||||||
    keyboard->keymap = keymap;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    char * xkb_keymap_str = xkb_keymap_get_as_string(keymap, XKB_KEYMAP_FORMAT_TEXT_V1);
 | 
					    char *xkb_keymap_str = xkb_keymap_get_as_string(keymap, XKB_KEYMAP_FORMAT_TEXT_V1);
 | 
				
			||||||
    keyboard->keymap_len = strlen(xkb_keymap_str) + 1;
 | 
					    size_t keymap_len = strlen(xkb_keymap_str) + 1;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    g_autofree char *path = strdup("/eek_keymap-XXXXXX");
 | 
					    g_autofree char *path = strdup("/eek_keymap-XXXXXX");
 | 
				
			||||||
    char *r = &path[strlen(path) - 6];
 | 
					    char *r = &path[strlen(path) - 6];
 | 
				
			||||||
@ -79,18 +69,42 @@ level_keyboard_new (struct squeek_layout *layout)
 | 
				
			|||||||
    if (keymap_fd < 0) {
 | 
					    if (keymap_fd < 0) {
 | 
				
			||||||
        g_error("Failed to set up keymap fd");
 | 
					        g_error("Failed to set up keymap fd");
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    keyboard->keymap_fd = keymap_fd;
 | 
					
 | 
				
			||||||
    shm_unlink(path);
 | 
					    shm_unlink(path);
 | 
				
			||||||
    if (ftruncate(keymap_fd, (off_t)keyboard->keymap_len)) {
 | 
					    if (ftruncate(keymap_fd, (off_t)keymap_len)) {
 | 
				
			||||||
        g_error("Failed to increase keymap fd size");
 | 
					        g_error("Failed to increase keymap fd size");
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    char *ptr = mmap(NULL, keyboard->keymap_len, PROT_WRITE, MAP_SHARED,
 | 
					    char *ptr = mmap(NULL, keymap_len, PROT_WRITE, MAP_SHARED,
 | 
				
			||||||
        keymap_fd, 0);
 | 
					        keymap_fd, 0);
 | 
				
			||||||
    if ((void*)ptr == (void*)-1) {
 | 
					    if ((void*)ptr == (void*)-1) {
 | 
				
			||||||
        g_error("Failed to set up mmap");
 | 
					        g_error("Failed to set up mmap");
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
    strncpy(ptr, xkb_keymap_str, keyboard->keymap_len);
 | 
					    strncpy(ptr, xkb_keymap_str, keymap_len);
 | 
				
			||||||
 | 
					    munmap(ptr, keymap_len);
 | 
				
			||||||
    free(xkb_keymap_str);
 | 
					    free(xkb_keymap_str);
 | 
				
			||||||
    munmap(ptr, keyboard->keymap_len);
 | 
					    xkb_keymap_unref(keymap);
 | 
				
			||||||
 | 
					    struct keymap km = {
 | 
				
			||||||
 | 
					        .fd = keymap_fd,
 | 
				
			||||||
 | 
					        .fd_len = keymap_len,
 | 
				
			||||||
 | 
					    };
 | 
				
			||||||
 | 
					    return km;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					void level_keyboard_free(LevelKeyboard *self) {
 | 
				
			||||||
 | 
					    eek_key_map_deinit(&self->keymap);
 | 
				
			||||||
 | 
					    squeek_layout_free(self->layout);
 | 
				
			||||||
 | 
					    g_free(self);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					LevelKeyboard*
 | 
				
			||||||
 | 
					level_keyboard_new (struct squeek_layout *layout)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					    LevelKeyboard *keyboard = g_new0(LevelKeyboard, 1);
 | 
				
			||||||
 | 
					    if (!keyboard) {
 | 
				
			||||||
 | 
					        g_error("Failed to create a keyboard");
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					    keyboard->layout = layout;
 | 
				
			||||||
 | 
					    const gchar *keymap_str = squeek_layout_get_keymap(keyboard->layout);
 | 
				
			||||||
 | 
					    keyboard->keymap = eek_key_map_from_str(keymap_str);
 | 
				
			||||||
    return keyboard;
 | 
					    return keyboard;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -32,19 +32,20 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
G_BEGIN_DECLS
 | 
					G_BEGIN_DECLS
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					/// Keymap container for Rust interoperability.
 | 
				
			||||||
 | 
					struct keymap {
 | 
				
			||||||
 | 
					    uint32_t fd; // keymap formatted as XKB string
 | 
				
			||||||
 | 
					    size_t fd_len; // length of the data inside keymap_fd
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
/// Keyboard state holder
 | 
					/// Keyboard state holder
 | 
				
			||||||
struct _LevelKeyboard {
 | 
					struct _LevelKeyboard {
 | 
				
			||||||
    struct squeek_layout *layout; // owned
 | 
					    struct squeek_layout *layout; // owned
 | 
				
			||||||
    struct xkb_keymap *keymap; // owned
 | 
					    struct keymap keymap; // owned
 | 
				
			||||||
    int keymap_fd; // keymap formatted as XKB string
 | 
					 | 
				
			||||||
    size_t keymap_len; // length of the data inside keymap_fd
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    guint id; // as a key to layout choices
 | 
					 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
typedef struct _LevelKeyboard LevelKeyboard;
 | 
					typedef struct _LevelKeyboard LevelKeyboard;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
gchar *             eek_keyboard_get_keymap
 | 
					gchar *eek_keyboard_get_keymap(LevelKeyboard *keyboard);
 | 
				
			||||||
                                     (LevelKeyboard *keyboard);
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
LevelKeyboard*
 | 
					LevelKeyboard*
 | 
				
			||||||
level_keyboard_new (struct squeek_layout *layout);
 | 
					level_keyboard_new (struct squeek_layout *layout);
 | 
				
			||||||
 | 
				
			|||||||
@ -17,7 +17,7 @@ eek_virtual_keyboard_v1_key(struct zwp_virtual_keyboard_v1 *zwp_virtual_keyboard
 | 
				
			|||||||
void eek_virtual_keyboard_update_keymap(struct zwp_virtual_keyboard_v1 *zwp_virtual_keyboard_v1, const LevelKeyboard *keyboard) {
 | 
					void eek_virtual_keyboard_update_keymap(struct zwp_virtual_keyboard_v1 *zwp_virtual_keyboard_v1, const LevelKeyboard *keyboard) {
 | 
				
			||||||
    zwp_virtual_keyboard_v1_keymap(zwp_virtual_keyboard_v1,
 | 
					    zwp_virtual_keyboard_v1_keymap(zwp_virtual_keyboard_v1,
 | 
				
			||||||
        WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1,
 | 
					        WL_KEYBOARD_KEYMAP_FORMAT_XKB_V1,
 | 
				
			||||||
        keyboard->keymap_fd, keyboard->keymap_len);
 | 
					        keyboard->keymap.fd, keyboard->keymap.fd_len);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void
 | 
					void
 | 
				
			||||||
 | 
				
			|||||||
		Reference in New Issue
	
	Block a user