mirror of
https://gitlab.freedesktop.org/wayland/wayland.git
synced 2025-11-04 13:29:51 -05:00
Add crude input device support.
Just pointer motion and button clicks for now, broadcast to all clients.
This commit is contained in:
parent
362a34ed89
commit
f9bc795a7d
6 changed files with 274 additions and 12 deletions
2
Makefile
2
Makefile
|
|
@ -3,7 +3,7 @@ LDLIBS += $(shell pkg-config --libs libffi libdrm)
|
||||||
|
|
||||||
all : wayland client
|
all : wayland client
|
||||||
|
|
||||||
wayland_objs = wayland.o event-loop.o connection.o hash.o egl-compositor.o
|
wayland_objs = wayland.o event-loop.o connection.o hash.o input.o egl-compositor.o
|
||||||
wayland : CFLAGS += -I../eagle
|
wayland : CFLAGS += -I../eagle
|
||||||
wayland : LDLIBS += -L../eagle -leagle -ldl
|
wayland : LDLIBS += -L../eagle -leagle -ldl
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -31,6 +31,9 @@ wl_event_loop_add_fd(struct wl_event_loop *loop,
|
||||||
struct epoll_event ep;
|
struct epoll_event ep;
|
||||||
|
|
||||||
source = malloc(sizeof *source);
|
source = malloc(sizeof *source);
|
||||||
|
if (source == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
source->fd = fd;
|
source->fd = fd;
|
||||||
source->func = func;
|
source->func = func;
|
||||||
source->data = data;
|
source->data = data;
|
||||||
|
|
|
||||||
152
input.c
Normal file
152
input.c
Normal file
|
|
@ -0,0 +1,152 @@
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <linux/input.h>
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <linux/input.h>
|
||||||
|
|
||||||
|
#include "wayland.h"
|
||||||
|
|
||||||
|
struct wl_input_device {
|
||||||
|
struct wl_object base;
|
||||||
|
struct wl_event_source *source;
|
||||||
|
struct wl_display *display;
|
||||||
|
int fd;
|
||||||
|
int tool;
|
||||||
|
int32_t x, y;
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct wl_method input_device_methods[] = {
|
||||||
|
};
|
||||||
|
|
||||||
|
static const struct wl_interface input_device_interface = {
|
||||||
|
"input_device", 1,
|
||||||
|
ARRAY_LENGTH(input_device_methods),
|
||||||
|
input_device_methods,
|
||||||
|
};
|
||||||
|
|
||||||
|
static void wl_input_device_data(int fd, uint32_t mask, void *data)
|
||||||
|
{
|
||||||
|
struct wl_input_device *device = data;
|
||||||
|
struct input_event ev[8], *e, *end;
|
||||||
|
int len, value, dx, dy, absolute_event;
|
||||||
|
|
||||||
|
dx = 0;
|
||||||
|
dy = 0;
|
||||||
|
absolute_event = 0;
|
||||||
|
|
||||||
|
len = read(fd, &ev, sizeof ev);
|
||||||
|
if (len < 0 || len % sizeof e[0] != 0) {
|
||||||
|
/* FIXME: handle error... reopen device? */;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
e = ev;
|
||||||
|
end = (void *) ev + len;
|
||||||
|
for (e = ev; e < end; e++) {
|
||||||
|
/* Get the signed value, earlier kernels had this as unsigned */
|
||||||
|
value = e->value;
|
||||||
|
|
||||||
|
switch (e->type) {
|
||||||
|
case EV_REL:
|
||||||
|
switch (e->code) {
|
||||||
|
case REL_X:
|
||||||
|
dx += value;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case REL_Y:
|
||||||
|
dy += value;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case EV_ABS:
|
||||||
|
absolute_event = 1;
|
||||||
|
switch (e->code) {
|
||||||
|
case ABS_X:
|
||||||
|
device->x = value;
|
||||||
|
break;
|
||||||
|
case ABS_Y:
|
||||||
|
device->y = value;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case EV_KEY:
|
||||||
|
if (value == 2)
|
||||||
|
break;
|
||||||
|
|
||||||
|
switch (e->code) {
|
||||||
|
case BTN_TOUCH:
|
||||||
|
case BTN_TOOL_PEN:
|
||||||
|
case BTN_TOOL_RUBBER:
|
||||||
|
case BTN_TOOL_BRUSH:
|
||||||
|
case BTN_TOOL_PENCIL:
|
||||||
|
case BTN_TOOL_AIRBRUSH:
|
||||||
|
case BTN_TOOL_FINGER:
|
||||||
|
case BTN_TOOL_MOUSE:
|
||||||
|
case BTN_TOOL_LENS:
|
||||||
|
device->tool = value ? e->code : 0;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case BTN_LEFT:
|
||||||
|
wl_display_post_button_event(device->display,
|
||||||
|
&device->base, 0, value);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case BTN_RIGHT:
|
||||||
|
wl_display_post_button_event(device->display,
|
||||||
|
&device->base, 2, value);
|
||||||
|
break;
|
||||||
|
|
||||||
|
case BTN_MIDDLE:
|
||||||
|
wl_display_post_button_event(device->display,
|
||||||
|
&device->base, 1, value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dx != 0 || dy != 0)
|
||||||
|
wl_display_post_relative_event(device->display,
|
||||||
|
&device->base, dx, dy);
|
||||||
|
if (absolute_event && device->tool)
|
||||||
|
wl_display_post_absolute_event(device->display,
|
||||||
|
&device->base,
|
||||||
|
device->x, device->y);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct wl_object *
|
||||||
|
wl_input_device_create(struct wl_display *display,
|
||||||
|
const char *path, uint32_t id)
|
||||||
|
{
|
||||||
|
struct wl_input_device *device;
|
||||||
|
struct wl_event_loop *loop;
|
||||||
|
|
||||||
|
device = malloc(sizeof *device);
|
||||||
|
if (device == NULL)
|
||||||
|
return NULL;
|
||||||
|
|
||||||
|
device->base.id = id;
|
||||||
|
device->base.interface = &input_device_interface;
|
||||||
|
device->display = display;
|
||||||
|
device->tool = 1;
|
||||||
|
|
||||||
|
device->fd = open(path, O_RDONLY);
|
||||||
|
if (device->fd < 0) {
|
||||||
|
free(device);
|
||||||
|
fprintf(stderr, "couldn't create pointer for %s: %m\n", path);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
loop = wl_display_get_event_loop(display);
|
||||||
|
device->source = wl_event_loop_add_fd(loop, device->fd,
|
||||||
|
WL_EVENT_READABLE,
|
||||||
|
wl_input_device_data, device);
|
||||||
|
if (device->source == NULL) {
|
||||||
|
close(device->fd);
|
||||||
|
free(device);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
|
return &device->base;
|
||||||
|
}
|
||||||
|
|
@ -96,16 +96,14 @@ wl_display_get_fd(struct wl_display *display)
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
handle_event(struct wl_connection *connection)
|
handle_event(struct wl_connection *connection, uint32_t opcode, uint32_t size)
|
||||||
{
|
{
|
||||||
uint32_t p[2], opcode, size;
|
uint32_t p[4];
|
||||||
|
|
||||||
wl_connection_copy(connection, p, sizeof p);
|
wl_connection_copy(connection, p, size);
|
||||||
opcode = p[1] & 0xffff;
|
printf("signal from object %d, opcode %d, size %d, args: %d, %d\n",
|
||||||
size = p[1] >> 16;
|
p[0], opcode, size, p[2], p[3]);
|
||||||
printf("signal from object %d, opcode %d, size %d\n",
|
wl_connection_consume(connection, size);
|
||||||
p[0], opcode, size);
|
|
||||||
wl_connection_consume(connection, sizeof p);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
|
|
@ -125,7 +123,8 @@ wl_display_iterate(struct wl_display *display, uint32_t mask)
|
||||||
if (len < size)
|
if (len < size)
|
||||||
break;
|
break;
|
||||||
|
|
||||||
handle_event(display->connection);
|
handle_event(display->connection, opcode, size);
|
||||||
|
len -= size;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (len < 0) {
|
if (len < 0) {
|
||||||
|
|
|
||||||
97
wayland.c
97
wayland.c
|
|
@ -12,8 +12,6 @@
|
||||||
#include "wayland.h"
|
#include "wayland.h"
|
||||||
#include "connection.h"
|
#include "connection.h"
|
||||||
|
|
||||||
#define ARRAY_LENGTH(a) (sizeof (a) / sizeof (a)[0])
|
|
||||||
|
|
||||||
#define container_of(ptr, type, member) ({ \
|
#define container_of(ptr, type, member) ({ \
|
||||||
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
|
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
|
||||||
(type *)( (char *)__mptr - offsetof(type,member) );})
|
(type *)( (char *)__mptr - offsetof(type,member) );})
|
||||||
|
|
@ -51,6 +49,7 @@ struct wl_client {
|
||||||
struct wl_event_source *source;
|
struct wl_event_source *source;
|
||||||
struct wl_display *display;
|
struct wl_display *display;
|
||||||
struct wl_list object_list;
|
struct wl_list object_list;
|
||||||
|
struct wl_list link;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct wl_display {
|
struct wl_display {
|
||||||
|
|
@ -58,11 +57,17 @@ struct wl_display {
|
||||||
struct wl_event_loop *loop;
|
struct wl_event_loop *loop;
|
||||||
struct wl_hash objects;
|
struct wl_hash objects;
|
||||||
|
|
||||||
|
struct wl_object *pointer;
|
||||||
|
|
||||||
struct wl_compositor *compositor;
|
struct wl_compositor *compositor;
|
||||||
struct wl_compositor_interface *compositor_interface;
|
struct wl_compositor_interface *compositor_interface;
|
||||||
|
|
||||||
struct wl_list surface_list;
|
struct wl_list surface_list;
|
||||||
|
struct wl_list client_list;
|
||||||
uint32_t client_id_range;
|
uint32_t client_id_range;
|
||||||
|
|
||||||
|
int32_t pointer_x;
|
||||||
|
int32_t pointer_y;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct wl_surface {
|
struct wl_surface {
|
||||||
|
|
@ -403,6 +408,8 @@ wl_client_create(struct wl_display *display, int fd)
|
||||||
|
|
||||||
advertise_object(client, &display->base);
|
advertise_object(client, &display->base);
|
||||||
|
|
||||||
|
wl_list_insert(display->client_list.prev, &client->link);
|
||||||
|
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -413,6 +420,8 @@ wl_client_destroy(struct wl_client *client)
|
||||||
|
|
||||||
printf("disconnect from client %p\n", client);
|
printf("disconnect from client %p\n", client);
|
||||||
|
|
||||||
|
wl_list_remove(&client->link);
|
||||||
|
|
||||||
while (client->object_list.next != &client->object_list) {
|
while (client->object_list.next != &client->object_list) {
|
||||||
ref = container_of(client->object_list.next,
|
ref = container_of(client->object_list.next,
|
||||||
struct wl_object_ref, link);
|
struct wl_object_ref, link);
|
||||||
|
|
@ -469,6 +478,21 @@ static const struct wl_interface display_interface = {
|
||||||
ARRAY_LENGTH(display_events), display_events,
|
ARRAY_LENGTH(display_events), display_events,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const char input_device_file[] =
|
||||||
|
"/dev/input/by-id/usb-Apple__Inc._Apple_Internal_Keyboard_._Trackpad-event-mouse";
|
||||||
|
|
||||||
|
static void
|
||||||
|
wl_display_create_input_devices(struct wl_display *display)
|
||||||
|
{
|
||||||
|
display->pointer = wl_input_device_create(display, input_device_file, 1);
|
||||||
|
|
||||||
|
if (display->pointer != NULL)
|
||||||
|
wl_hash_insert(&display->objects, display->pointer);
|
||||||
|
|
||||||
|
display->pointer_x = 100;
|
||||||
|
display->pointer_y = 100;
|
||||||
|
}
|
||||||
|
|
||||||
struct wl_display *
|
struct wl_display *
|
||||||
wl_display_create(void)
|
wl_display_create(void)
|
||||||
{
|
{
|
||||||
|
|
@ -488,12 +512,81 @@ wl_display_create(void)
|
||||||
display->base.interface = &display_interface;
|
display->base.interface = &display_interface;
|
||||||
wl_hash_insert(&display->objects, &display->base);
|
wl_hash_insert(&display->objects, &display->base);
|
||||||
wl_list_init(&display->surface_list);
|
wl_list_init(&display->surface_list);
|
||||||
|
wl_list_init(&display->client_list);
|
||||||
|
|
||||||
|
wl_display_create_input_devices(display);
|
||||||
|
|
||||||
display->client_id_range = 256; /* Gah, arbitrary... */
|
display->client_id_range = 256; /* Gah, arbitrary... */
|
||||||
|
|
||||||
return display;
|
return display;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
wl_display_send_event(struct wl_display *display, uint32_t *data, size_t size)
|
||||||
|
{
|
||||||
|
struct wl_client *client;
|
||||||
|
|
||||||
|
client = container_of(display->client_list.next,
|
||||||
|
struct wl_client, link);
|
||||||
|
while (&client->link != &display->client_list) {
|
||||||
|
wl_connection_write(client->connection, data, size);
|
||||||
|
|
||||||
|
client = container_of(client->link.next,
|
||||||
|
struct wl_client, link);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#define WL_POINTER_MOTION 0
|
||||||
|
#define WL_POINTER_BUTTON 1
|
||||||
|
|
||||||
|
void
|
||||||
|
wl_display_post_relative_event(struct wl_display *display,
|
||||||
|
struct wl_object *source, int dx, int dy)
|
||||||
|
{
|
||||||
|
uint32_t p[4];
|
||||||
|
|
||||||
|
display->pointer_x += dx;
|
||||||
|
display->pointer_y += dy;
|
||||||
|
|
||||||
|
p[0] = source->id;
|
||||||
|
p[1] = (sizeof p << 16) | WL_POINTER_MOTION;
|
||||||
|
p[2] = display->pointer_x;
|
||||||
|
p[3] = display->pointer_y;
|
||||||
|
|
||||||
|
wl_display_send_event(display, p, sizeof p);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
wl_display_post_absolute_event(struct wl_display *display,
|
||||||
|
struct wl_object *source, int x, int y)
|
||||||
|
{
|
||||||
|
uint32_t p[4];
|
||||||
|
|
||||||
|
display->pointer_x = x;
|
||||||
|
display->pointer_y = y;
|
||||||
|
|
||||||
|
p[0] = source->id;
|
||||||
|
p[1] = (sizeof p << 16) | WL_POINTER_MOTION;
|
||||||
|
p[2] = display->pointer_x;
|
||||||
|
p[3] = display->pointer_y;
|
||||||
|
|
||||||
|
wl_display_send_event(display, p, sizeof p);
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
wl_display_post_button_event(struct wl_display *display,
|
||||||
|
struct wl_object *source, int button, int state)
|
||||||
|
{
|
||||||
|
uint32_t p[4];
|
||||||
|
|
||||||
|
p[0] = source->id;
|
||||||
|
p[1] = (sizeof p << 16) | WL_POINTER_BUTTON;
|
||||||
|
p[2] = button;
|
||||||
|
p[3] = state;
|
||||||
|
|
||||||
|
wl_display_send_event(display, p, sizeof p);
|
||||||
|
}
|
||||||
|
|
||||||
void
|
void
|
||||||
wl_display_set_compositor(struct wl_display *display,
|
wl_display_set_compositor(struct wl_display *display,
|
||||||
struct wl_compositor *compositor)
|
struct wl_compositor *compositor)
|
||||||
|
|
|
||||||
15
wayland.h
15
wayland.h
|
|
@ -3,6 +3,8 @@
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
|
|
||||||
|
#define ARRAY_LENGTH(a) (sizeof (a) / sizeof (a)[0])
|
||||||
|
|
||||||
enum {
|
enum {
|
||||||
WL_EVENT_READABLE = 0x01,
|
WL_EVENT_READABLE = 0x01,
|
||||||
WL_EVENT_WRITEABLE = 0x02
|
WL_EVENT_WRITEABLE = 0x02
|
||||||
|
|
@ -97,6 +99,19 @@ int wl_surface_iterator_next(struct wl_surface_iterator *iterator,
|
||||||
struct wl_surface **surface);
|
struct wl_surface **surface);
|
||||||
void wl_surface_iterator_destroy(struct wl_surface_iterator *iterator);
|
void wl_surface_iterator_destroy(struct wl_surface_iterator *iterator);
|
||||||
|
|
||||||
|
struct wl_object *
|
||||||
|
wl_input_device_create(struct wl_display *display,
|
||||||
|
const char *path, uint32_t id);
|
||||||
|
void
|
||||||
|
wl_display_post_relative_event(struct wl_display *display,
|
||||||
|
struct wl_object *source, int dx, int dy);
|
||||||
|
void
|
||||||
|
wl_display_post_absolute_event(struct wl_display *display,
|
||||||
|
struct wl_object *source, int x, int y);
|
||||||
|
void
|
||||||
|
wl_display_post_button_event(struct wl_display *display,
|
||||||
|
struct wl_object *source, int button, int state);
|
||||||
|
|
||||||
struct wl_compositor {
|
struct wl_compositor {
|
||||||
struct wl_compositor_interface *interface;
|
struct wl_compositor_interface *interface;
|
||||||
};
|
};
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue