Add crude input device support.

Just pointer motion and button clicks for now, broadcast to all clients.
This commit is contained in:
Kristian Høgsberg 2008-11-02 10:12:29 -05:00
parent 362a34ed89
commit f9bc795a7d
6 changed files with 274 additions and 12 deletions

View File

@ -3,7 +3,7 @@ LDLIBS += $(shell pkg-config --libs libffi libdrm)
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 : LDLIBS += -L../eagle -leagle -ldl

View File

@ -31,6 +31,9 @@ wl_event_loop_add_fd(struct wl_event_loop *loop,
struct epoll_event ep;
source = malloc(sizeof *source);
if (source == NULL)
return NULL;
source->fd = fd;
source->func = func;
source->data = data;

152
input.c Normal file
View 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;
}

View File

@ -96,16 +96,14 @@ wl_display_get_fd(struct wl_display *display)
}
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);
opcode = p[1] & 0xffff;
size = p[1] >> 16;
printf("signal from object %d, opcode %d, size %d\n",
p[0], opcode, size);
wl_connection_consume(connection, sizeof p);
wl_connection_copy(connection, p, size);
printf("signal from object %d, opcode %d, size %d, args: %d, %d\n",
p[0], opcode, size, p[2], p[3]);
wl_connection_consume(connection, size);
}
void
@ -125,7 +123,8 @@ wl_display_iterate(struct wl_display *display, uint32_t mask)
if (len < size)
break;
handle_event(display->connection);
handle_event(display->connection, opcode, size);
len -= size;
}
if (len < 0) {

View File

@ -12,8 +12,6 @@
#include "wayland.h"
#include "connection.h"
#define ARRAY_LENGTH(a) (sizeof (a) / sizeof (a)[0])
#define container_of(ptr, type, member) ({ \
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type,member) );})
@ -51,6 +49,7 @@ struct wl_client {
struct wl_event_source *source;
struct wl_display *display;
struct wl_list object_list;
struct wl_list link;
};
struct wl_display {
@ -58,11 +57,17 @@ struct wl_display {
struct wl_event_loop *loop;
struct wl_hash objects;
struct wl_object *pointer;
struct wl_compositor *compositor;
struct wl_compositor_interface *compositor_interface;
struct wl_list surface_list;
struct wl_list client_list;
uint32_t client_id_range;
int32_t pointer_x;
int32_t pointer_y;
};
struct wl_surface {
@ -403,6 +408,8 @@ wl_client_create(struct wl_display *display, int fd)
advertise_object(client, &display->base);
wl_list_insert(display->client_list.prev, &client->link);
return client;
}
@ -413,6 +420,8 @@ wl_client_destroy(struct wl_client *client)
printf("disconnect from client %p\n", client);
wl_list_remove(&client->link);
while (client->object_list.next != &client->object_list) {
ref = container_of(client->object_list.next,
struct wl_object_ref, link);
@ -469,6 +478,21 @@ static const struct wl_interface display_interface = {
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 *
wl_display_create(void)
{
@ -488,12 +512,81 @@ wl_display_create(void)
display->base.interface = &display_interface;
wl_hash_insert(&display->objects, &display->base);
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... */
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
wl_display_set_compositor(struct wl_display *display,
struct wl_compositor *compositor)

View File

@ -3,6 +3,8 @@
#include <stdint.h>
#define ARRAY_LENGTH(a) (sizeof (a) / sizeof (a)[0])
enum {
WL_EVENT_READABLE = 0x01,
WL_EVENT_WRITEABLE = 0x02
@ -97,6 +99,19 @@ int wl_surface_iterator_next(struct wl_surface_iterator *iterator,
struct wl_surface **surface);
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_interface *interface;
};