2008-12-02 15:15:01 -05:00
|
|
|
/*
|
|
|
|
|
* Copyright © 2008 Kristian Høgsberg
|
|
|
|
|
*
|
|
|
|
|
* Permission to use, copy, modify, distribute, and sell this software and its
|
|
|
|
|
* documentation for any purpose is hereby granted without fee, provided that
|
|
|
|
|
* the above copyright notice appear in all copies and that both that copyright
|
|
|
|
|
* notice and this permission notice appear in supporting documentation, and
|
|
|
|
|
* that the name of the copyright holders not be used in advertising or
|
|
|
|
|
* publicity pertaining to distribution of the software without specific,
|
|
|
|
|
* written prior permission. The copyright holders make no representations
|
|
|
|
|
* about the suitability of this software for any purpose. It is provided "as
|
|
|
|
|
* is" without express or implied warranty.
|
|
|
|
|
*
|
|
|
|
|
* THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
|
|
|
|
|
* INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
|
|
|
|
|
* EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
|
|
|
|
|
* CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
|
|
|
|
|
* DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
|
|
|
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
|
|
|
|
|
* OF THIS SOFTWARE.
|
|
|
|
|
*/
|
|
|
|
|
|
2008-10-07 10:10:36 -04:00
|
|
|
#include <stdlib.h>
|
|
|
|
|
#include <stdint.h>
|
|
|
|
|
#include <stddef.h>
|
|
|
|
|
#include <stdio.h>
|
2011-07-14 18:56:40 +03:00
|
|
|
#include <stdbool.h>
|
2008-10-07 10:10:36 -04:00
|
|
|
#include <errno.h>
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#include <sys/socket.h>
|
|
|
|
|
#include <sys/un.h>
|
|
|
|
|
#include <ctype.h>
|
2008-12-21 21:50:23 -05:00
|
|
|
#include <assert.h>
|
2011-04-11 09:24:11 -04:00
|
|
|
#include <fcntl.h>
|
2008-10-07 10:10:36 -04:00
|
|
|
#include <sys/poll.h>
|
|
|
|
|
|
2008-10-08 13:32:07 -04:00
|
|
|
#include "connection.h"
|
2008-11-23 23:41:08 -05:00
|
|
|
#include "wayland-util.h"
|
2008-10-08 13:32:07 -04:00
|
|
|
#include "wayland-client.h"
|
|
|
|
|
|
2008-12-30 11:03:33 -05:00
|
|
|
struct wl_global_listener {
|
|
|
|
|
wl_display_global_func_t handler;
|
|
|
|
|
void *data;
|
|
|
|
|
struct wl_list link;
|
2008-10-07 10:10:36 -04:00
|
|
|
};
|
|
|
|
|
|
2008-12-30 11:03:33 -05:00
|
|
|
struct wl_proxy {
|
2010-12-01 17:07:41 -05:00
|
|
|
struct wl_object object;
|
2008-12-30 11:03:33 -05:00
|
|
|
struct wl_display *display;
|
2009-09-18 09:49:21 -04:00
|
|
|
void *user_data;
|
2008-10-07 10:10:36 -04:00
|
|
|
};
|
|
|
|
|
|
2011-04-14 10:38:44 -04:00
|
|
|
struct wl_global {
|
|
|
|
|
uint32_t id;
|
|
|
|
|
char *interface;
|
|
|
|
|
uint32_t version;
|
|
|
|
|
struct wl_list link;
|
|
|
|
|
};
|
|
|
|
|
|
2008-12-30 11:03:33 -05:00
|
|
|
struct wl_display {
|
|
|
|
|
struct wl_proxy proxy;
|
|
|
|
|
struct wl_connection *connection;
|
|
|
|
|
int fd;
|
|
|
|
|
uint32_t id, id_count, next_range;
|
|
|
|
|
uint32_t mask;
|
2010-02-26 11:42:59 -05:00
|
|
|
struct wl_hash_table *objects;
|
2008-12-30 11:03:33 -05:00
|
|
|
struct wl_list global_listener_list;
|
2011-04-14 10:38:44 -04:00
|
|
|
struct wl_list global_list;
|
2008-12-30 11:03:33 -05:00
|
|
|
|
|
|
|
|
wl_display_update_func_t update;
|
|
|
|
|
void *update_data;
|
|
|
|
|
|
|
|
|
|
wl_display_global_func_t global_handler;
|
|
|
|
|
void *global_handler_data;
|
|
|
|
|
};
|
|
|
|
|
|
2011-02-10 12:27:35 -05:00
|
|
|
static int wl_debug = 0;
|
|
|
|
|
|
2008-11-07 14:27:23 -05:00
|
|
|
static int
|
|
|
|
|
connection_update(struct wl_connection *connection,
|
|
|
|
|
uint32_t mask, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct wl_display *display = data;
|
|
|
|
|
|
|
|
|
|
display->mask = mask;
|
|
|
|
|
if (display->update)
|
|
|
|
|
return display->update(display->mask,
|
|
|
|
|
display->update_data);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-30 11:03:33 -05:00
|
|
|
WL_EXPORT struct wl_global_listener *
|
|
|
|
|
wl_display_add_global_listener(struct wl_display *display,
|
|
|
|
|
wl_display_global_func_t handler, void *data)
|
|
|
|
|
{
|
|
|
|
|
struct wl_global_listener *listener;
|
2011-06-10 10:27:16 +02:00
|
|
|
struct wl_global *global;
|
2008-12-24 19:30:25 -05:00
|
|
|
|
2008-12-30 11:03:33 -05:00
|
|
|
listener = malloc(sizeof *listener);
|
|
|
|
|
if (listener == NULL)
|
|
|
|
|
return NULL;
|
2008-12-24 19:30:25 -05:00
|
|
|
|
2008-12-30 11:03:33 -05:00
|
|
|
listener->handler = handler;
|
|
|
|
|
listener->data = data;
|
|
|
|
|
wl_list_insert(display->global_listener_list.prev, &listener->link);
|
2008-12-24 19:30:25 -05:00
|
|
|
|
2011-06-10 10:27:16 +02:00
|
|
|
wl_list_for_each(global, &display->global_list, link)
|
|
|
|
|
(*listener->handler)(display, global->id, global->interface,
|
|
|
|
|
global->version, listener->data);
|
|
|
|
|
|
2008-12-30 11:03:33 -05:00
|
|
|
return listener;
|
2008-12-24 19:30:25 -05:00
|
|
|
}
|
|
|
|
|
|
2008-12-21 23:37:12 -05:00
|
|
|
WL_EXPORT void
|
2008-12-30 11:03:33 -05:00
|
|
|
wl_display_remove_global_listener(struct wl_display *display,
|
|
|
|
|
struct wl_global_listener *listener)
|
|
|
|
|
{
|
|
|
|
|
wl_list_remove(&listener->link);
|
|
|
|
|
free(listener);
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-10 14:02:48 -04:00
|
|
|
WL_EXPORT struct wl_proxy *
|
|
|
|
|
wl_proxy_create_for_id(struct wl_display *display,
|
|
|
|
|
const struct wl_interface *interface, uint32_t id)
|
2008-12-30 11:03:33 -05:00
|
|
|
{
|
|
|
|
|
struct wl_proxy *proxy;
|
|
|
|
|
|
|
|
|
|
proxy = malloc(sizeof *proxy);
|
|
|
|
|
if (proxy == NULL)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
2010-12-01 17:07:41 -05:00
|
|
|
proxy->object.interface = interface;
|
2011-02-18 15:28:54 -05:00
|
|
|
proxy->object.implementation = NULL;
|
2010-12-01 17:07:41 -05:00
|
|
|
proxy->object.id = id;
|
2008-12-30 11:03:33 -05:00
|
|
|
proxy->display = display;
|
2010-12-01 17:07:41 -05:00
|
|
|
wl_hash_table_insert(display->objects, proxy->object.id, proxy);
|
2008-12-30 11:03:33 -05:00
|
|
|
|
|
|
|
|
return proxy;
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-09 21:25:50 -04:00
|
|
|
WL_EXPORT struct wl_proxy *
|
2010-08-10 14:02:48 -04:00
|
|
|
wl_proxy_create(struct wl_proxy *factory,
|
|
|
|
|
const struct wl_interface *interface)
|
2010-08-09 21:25:50 -04:00
|
|
|
{
|
2010-08-10 14:02:48 -04:00
|
|
|
return wl_proxy_create_for_id(factory->display, interface,
|
|
|
|
|
wl_display_allocate_id(factory->display));
|
2010-08-09 21:25:50 -04:00
|
|
|
}
|
|
|
|
|
|
2010-09-02 20:22:42 -04:00
|
|
|
WL_EXPORT void
|
|
|
|
|
wl_proxy_destroy(struct wl_proxy *proxy)
|
|
|
|
|
{
|
2010-12-01 17:07:41 -05:00
|
|
|
wl_hash_table_remove(proxy->display->objects, proxy->object.id);
|
2010-09-02 20:22:42 -04:00
|
|
|
free(proxy);
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-10 10:53:44 -04:00
|
|
|
WL_EXPORT int
|
2010-08-10 14:02:48 -04:00
|
|
|
wl_proxy_add_listener(struct wl_proxy *proxy,
|
|
|
|
|
void (**implementation)(void), void *data)
|
2008-12-21 23:37:12 -05:00
|
|
|
{
|
2011-02-18 15:28:54 -05:00
|
|
|
if (proxy->object.implementation) {
|
|
|
|
|
fprintf(stderr, "proxy already has listener\n");
|
2008-12-30 11:03:33 -05:00
|
|
|
return -1;
|
2011-02-18 15:28:54 -05:00
|
|
|
}
|
2008-12-30 11:03:33 -05:00
|
|
|
|
2011-02-18 15:28:54 -05:00
|
|
|
proxy->object.implementation = implementation;
|
|
|
|
|
proxy->user_data = data;
|
2008-12-30 11:03:33 -05:00
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2010-08-09 21:25:50 -04:00
|
|
|
WL_EXPORT void
|
2008-12-30 11:03:33 -05:00
|
|
|
wl_proxy_marshal(struct wl_proxy *proxy, uint32_t opcode, ...)
|
|
|
|
|
{
|
2010-09-07 21:34:45 -04:00
|
|
|
struct wl_closure *closure;
|
2008-12-30 11:03:33 -05:00
|
|
|
va_list ap;
|
|
|
|
|
|
|
|
|
|
va_start(ap, opcode);
|
2010-09-07 21:34:45 -04:00
|
|
|
closure = wl_connection_vmarshal(proxy->display->connection,
|
2010-12-01 17:07:41 -05:00
|
|
|
&proxy->object, opcode, ap,
|
|
|
|
|
&proxy->object.interface->methods[opcode]);
|
2008-12-30 11:03:33 -05:00
|
|
|
va_end(ap);
|
2010-09-07 21:34:45 -04:00
|
|
|
|
|
|
|
|
wl_closure_send(closure, proxy->display->connection);
|
2011-02-10 12:27:35 -05:00
|
|
|
|
2011-07-14 18:56:40 +03:00
|
|
|
if (wl_debug)
|
|
|
|
|
wl_closure_print(closure, &proxy->object, true);
|
2011-02-10 12:27:35 -05:00
|
|
|
|
2010-09-07 21:34:45 -04:00
|
|
|
wl_closure_destroy(closure);
|
2008-12-30 11:03:33 -05:00
|
|
|
}
|
|
|
|
|
|
2011-04-14 10:38:44 -04:00
|
|
|
/* Can't do this, there may be more than one instance of an
|
|
|
|
|
* interface... */
|
|
|
|
|
WL_EXPORT uint32_t
|
|
|
|
|
wl_display_get_global(struct wl_display *display,
|
|
|
|
|
const char *interface, uint32_t version)
|
|
|
|
|
{
|
|
|
|
|
struct wl_global *global;
|
|
|
|
|
|
|
|
|
|
wl_list_for_each(global, &display->global_list, link)
|
|
|
|
|
if (strcmp(interface, global->interface) == 0 &&
|
|
|
|
|
version <= global->version)
|
|
|
|
|
return global->id;
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-24 19:30:25 -05:00
|
|
|
static void
|
2011-05-11 10:57:06 -04:00
|
|
|
display_handle_error(void *data,
|
|
|
|
|
struct wl_display *display, struct wl_object *object,
|
|
|
|
|
uint32_t code, const char *message)
|
2008-12-24 19:30:25 -05:00
|
|
|
{
|
2011-05-11 10:57:06 -04:00
|
|
|
fprintf(stderr, "%s@%d: error %d: %s\n",
|
|
|
|
|
object->interface->name, object->id, code, message);
|
2010-08-05 17:44:31 -04:00
|
|
|
abort();
|
2008-12-24 19:30:25 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
2008-12-30 11:03:33 -05:00
|
|
|
display_handle_global(void *data,
|
|
|
|
|
struct wl_display *display,
|
2008-12-24 19:30:25 -05:00
|
|
|
uint32_t id, const char *interface, uint32_t version)
|
|
|
|
|
{
|
2010-08-10 14:02:48 -04:00
|
|
|
struct wl_global_listener *listener;
|
2011-04-14 10:38:44 -04:00
|
|
|
struct wl_global *global;
|
2008-12-30 11:03:33 -05:00
|
|
|
|
2011-04-18 10:24:11 -04:00
|
|
|
if (strcmp(interface, "wl_display") == 0)
|
2010-02-26 11:42:59 -05:00
|
|
|
wl_hash_table_insert(display->objects,
|
2010-12-01 17:07:41 -05:00
|
|
|
id, &display->proxy.object);
|
2010-08-10 14:02:48 -04:00
|
|
|
|
2011-04-14 10:38:44 -04:00
|
|
|
global = malloc(sizeof *global);
|
|
|
|
|
global->id = id;
|
|
|
|
|
global->interface = strdup(interface);
|
|
|
|
|
global->version = version;
|
|
|
|
|
wl_list_insert(display->global_list.prev, &global->link);
|
|
|
|
|
|
2010-08-10 14:02:48 -04:00
|
|
|
wl_list_for_each(listener, &display->global_listener_list, link)
|
|
|
|
|
(*listener->handler)(display,
|
|
|
|
|
id, interface, version, listener->data);
|
2008-12-24 19:30:25 -05:00
|
|
|
}
|
|
|
|
|
|
2011-06-14 10:35:46 +02:00
|
|
|
static void
|
|
|
|
|
display_handle_global_remove(void *data,
|
|
|
|
|
struct wl_display *display, uint32_t id)
|
|
|
|
|
{
|
|
|
|
|
struct wl_global *global;
|
|
|
|
|
|
|
|
|
|
wl_list_for_each(global, &display->global_list, link)
|
|
|
|
|
if (global->id == id) {
|
|
|
|
|
wl_list_remove(&global->link);
|
|
|
|
|
free(global);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2008-12-24 19:30:25 -05:00
|
|
|
static void
|
2008-12-30 11:03:33 -05:00
|
|
|
display_handle_range(void *data,
|
|
|
|
|
struct wl_display *display, uint32_t range)
|
2008-12-24 19:30:25 -05:00
|
|
|
{
|
|
|
|
|
display->next_range = range;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static const struct wl_display_listener display_listener = {
|
2011-05-11 10:57:06 -04:00
|
|
|
display_handle_error,
|
2008-12-24 19:30:25 -05:00
|
|
|
display_handle_global,
|
2011-06-14 10:35:46 +02:00
|
|
|
display_handle_global_remove,
|
2010-09-03 14:46:38 -04:00
|
|
|
display_handle_range,
|
2008-12-24 19:30:25 -05:00
|
|
|
};
|
|
|
|
|
|
2011-04-11 09:14:43 -04:00
|
|
|
static int
|
|
|
|
|
connect_to_socket(struct wl_display *display, const char *name)
|
2008-10-07 10:10:36 -04:00
|
|
|
{
|
2008-12-07 15:22:22 -05:00
|
|
|
struct sockaddr_un addr;
|
2008-10-07 10:10:36 -04:00
|
|
|
socklen_t size;
|
2010-12-01 15:36:20 -05:00
|
|
|
const char *runtime_dir;
|
|
|
|
|
size_t name_size;
|
2008-10-07 10:10:36 -04:00
|
|
|
|
2011-04-11 09:24:11 -04:00
|
|
|
display->fd = socket(PF_LOCAL, SOCK_STREAM | SOCK_CLOEXEC, 0);
|
2011-04-11 09:14:43 -04:00
|
|
|
if (display->fd < 0)
|
|
|
|
|
return -1;
|
2008-10-07 10:10:36 -04:00
|
|
|
|
2010-12-01 15:36:20 -05:00
|
|
|
runtime_dir = getenv("XDG_RUNTIME_DIR");
|
|
|
|
|
if (runtime_dir == NULL) {
|
|
|
|
|
runtime_dir = ".";
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"XDG_RUNTIME_DIR not set, falling back to %s\n",
|
|
|
|
|
runtime_dir);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (name == NULL)
|
|
|
|
|
name = getenv("WAYLAND_DISPLAY");
|
|
|
|
|
if (name == NULL)
|
|
|
|
|
name = "wayland-0";
|
|
|
|
|
|
|
|
|
|
memset(&addr, 0, sizeof addr);
|
2008-12-07 15:22:22 -05:00
|
|
|
addr.sun_family = AF_LOCAL;
|
2010-12-01 15:36:20 -05:00
|
|
|
name_size =
|
|
|
|
|
snprintf(addr.sun_path, sizeof addr.sun_path,
|
|
|
|
|
"%s/%s", runtime_dir, name) + 1;
|
2008-10-07 10:10:36 -04:00
|
|
|
|
2008-12-07 15:22:22 -05:00
|
|
|
size = offsetof (struct sockaddr_un, sun_path) + name_size;
|
2008-10-07 10:10:36 -04:00
|
|
|
|
2008-12-07 15:22:22 -05:00
|
|
|
if (connect(display->fd, (struct sockaddr *) &addr, size) < 0) {
|
2008-10-08 13:32:07 -04:00
|
|
|
close(display->fd);
|
2011-04-11 09:14:43 -04:00
|
|
|
return -1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
WL_EXPORT struct wl_display *
|
|
|
|
|
wl_display_connect(const char *name)
|
|
|
|
|
{
|
|
|
|
|
struct wl_display *display;
|
|
|
|
|
const char *debug;
|
|
|
|
|
char *connection, *end;
|
|
|
|
|
int flags;
|
|
|
|
|
|
|
|
|
|
debug = getenv("WAYLAND_DEBUG");
|
|
|
|
|
if (debug)
|
|
|
|
|
wl_debug = 1;
|
|
|
|
|
|
|
|
|
|
display = malloc(sizeof *display);
|
|
|
|
|
if (display == NULL)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
memset(display, 0, sizeof *display);
|
|
|
|
|
connection = getenv("WAYLAND_SOCKET");
|
|
|
|
|
if (connection) {
|
|
|
|
|
display->fd = strtol(connection, &end, 0);
|
|
|
|
|
if (*end != '\0') {
|
|
|
|
|
free(display);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
flags = fcntl(display->fd, F_GETFD);
|
|
|
|
|
if (flags != -1)
|
|
|
|
|
fcntl(display->fd, F_SETFD, flags | FD_CLOEXEC);
|
|
|
|
|
} else if (connect_to_socket(display, name) < 0) {
|
2008-10-08 13:32:07 -04:00
|
|
|
free(display);
|
2008-10-07 10:10:36 -04:00
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2010-02-26 11:42:59 -05:00
|
|
|
display->objects = wl_hash_table_create();
|
2011-03-11 14:58:06 +02:00
|
|
|
if (display->objects == NULL) {
|
|
|
|
|
close(display->fd);
|
|
|
|
|
free(display);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2008-12-30 11:03:33 -05:00
|
|
|
wl_list_init(&display->global_listener_list);
|
2011-04-14 10:38:44 -04:00
|
|
|
wl_list_init(&display->global_list);
|
2008-10-07 10:10:36 -04:00
|
|
|
|
2010-12-01 17:07:41 -05:00
|
|
|
display->proxy.object.interface = &wl_display_interface;
|
|
|
|
|
display->proxy.object.id = 1;
|
2008-12-21 21:50:23 -05:00
|
|
|
display->proxy.display = display;
|
2008-12-30 11:03:33 -05:00
|
|
|
|
2011-02-18 15:28:54 -05:00
|
|
|
display->proxy.object.implementation =
|
|
|
|
|
(void(**)(void)) &display_listener;
|
|
|
|
|
display->proxy.user_data = display;
|
2008-10-07 10:10:36 -04:00
|
|
|
|
2008-10-08 13:32:07 -04:00
|
|
|
display->connection = wl_connection_create(display->fd,
|
2008-11-07 14:27:23 -05:00
|
|
|
connection_update,
|
|
|
|
|
display);
|
2011-03-11 14:43:10 +02:00
|
|
|
if (display->connection == NULL) {
|
|
|
|
|
wl_hash_table_destroy(display->objects);
|
|
|
|
|
close(display->fd);
|
|
|
|
|
free(display);
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
2011-04-14 10:38:44 -04:00
|
|
|
|
2011-04-18 10:24:11 -04:00
|
|
|
wl_display_bind(display, 1, "wl_display", 1);
|
2011-04-14 10:38:44 -04:00
|
|
|
|
2008-10-08 13:32:07 -04:00
|
|
|
return display;
|
2008-10-07 10:10:36 -04:00
|
|
|
}
|
|
|
|
|
|
2008-11-08 15:39:41 -05:00
|
|
|
WL_EXPORT void
|
2008-10-08 13:32:07 -04:00
|
|
|
wl_display_destroy(struct wl_display *display)
|
2008-10-07 10:10:36 -04:00
|
|
|
{
|
2011-06-14 11:41:54 +02:00
|
|
|
struct wl_global *global, *gnext;
|
|
|
|
|
struct wl_global_listener *listener, *lnext;
|
|
|
|
|
|
2008-10-08 13:32:07 -04:00
|
|
|
wl_connection_destroy(display->connection);
|
2011-03-11 14:58:06 +02:00
|
|
|
wl_hash_table_destroy(display->objects);
|
2011-06-14 11:41:54 +02:00
|
|
|
wl_list_for_each_safe(global, gnext,
|
|
|
|
|
&display->global_list, link)
|
|
|
|
|
free(global);
|
|
|
|
|
wl_list_for_each_safe(listener, lnext,
|
|
|
|
|
&display->global_listener_list, link)
|
|
|
|
|
free(listener);
|
|
|
|
|
|
2008-10-08 13:32:07 -04:00
|
|
|
close(display->fd);
|
|
|
|
|
free(display);
|
2008-10-07 10:10:36 -04:00
|
|
|
}
|
|
|
|
|
|
2008-11-08 15:39:41 -05:00
|
|
|
WL_EXPORT int
|
2008-11-07 14:27:23 -05:00
|
|
|
wl_display_get_fd(struct wl_display *display,
|
|
|
|
|
wl_display_update_func_t update, void *data)
|
2008-10-07 10:10:36 -04:00
|
|
|
{
|
2008-11-07 14:27:23 -05:00
|
|
|
display->update = update;
|
|
|
|
|
display->update_data = data;
|
|
|
|
|
|
|
|
|
|
display->update(display->mask, display->update_data);
|
|
|
|
|
|
2008-10-08 13:32:07 -04:00
|
|
|
return display->fd;
|
2008-10-07 10:10:36 -04:00
|
|
|
}
|
|
|
|
|
|
2011-07-29 19:51:22 -07:00
|
|
|
static void
|
|
|
|
|
sync_callback(void *data, struct wl_callback *callback, uint32_t time)
|
2010-09-03 14:46:38 -04:00
|
|
|
{
|
2011-07-29 19:51:22 -07:00
|
|
|
int *done = data;
|
2010-09-03 14:46:38 -04:00
|
|
|
|
2011-07-29 19:51:22 -07:00
|
|
|
*done = 1;
|
|
|
|
|
wl_callback_destroy(callback);
|
2010-09-03 14:46:38 -04:00
|
|
|
}
|
|
|
|
|
|
2011-07-29 19:51:22 -07:00
|
|
|
static const struct wl_callback_listener sync_listener = {
|
|
|
|
|
sync_callback
|
|
|
|
|
};
|
2010-09-03 14:46:38 -04:00
|
|
|
|
2011-07-29 19:51:22 -07:00
|
|
|
WL_EXPORT void
|
|
|
|
|
wl_display_roundtrip(struct wl_display *display)
|
|
|
|
|
{
|
|
|
|
|
struct wl_callback *callback;
|
|
|
|
|
int done;
|
|
|
|
|
|
|
|
|
|
done = 0;
|
|
|
|
|
callback = wl_display_sync(display);
|
|
|
|
|
wl_callback_add_listener(callback, &sync_listener, &done);
|
|
|
|
|
wl_display_flush(display);
|
|
|
|
|
while (!done)
|
|
|
|
|
wl_display_iterate(display, WL_DISPLAY_READABLE);
|
2010-09-03 14:46:38 -04:00
|
|
|
}
|
|
|
|
|
|
2008-10-07 10:10:36 -04:00
|
|
|
static void
|
2008-11-25 22:40:39 -05:00
|
|
|
handle_event(struct wl_display *display,
|
2008-12-22 18:06:49 -05:00
|
|
|
uint32_t id, uint32_t opcode, uint32_t size)
|
2008-10-07 10:10:36 -04:00
|
|
|
{
|
2008-12-21 20:25:16 -05:00
|
|
|
uint32_t p[32];
|
2008-12-30 11:03:33 -05:00
|
|
|
struct wl_proxy *proxy;
|
2010-08-30 09:47:36 -04:00
|
|
|
struct wl_closure *closure;
|
2010-09-01 17:18:33 -04:00
|
|
|
const struct wl_message *message;
|
2008-10-07 10:10:36 -04:00
|
|
|
|
2008-11-02 10:55:25 -05:00
|
|
|
wl_connection_copy(display->connection, p, size);
|
2008-12-24 19:30:25 -05:00
|
|
|
if (id == 1)
|
2008-12-30 11:03:33 -05:00
|
|
|
proxy = &display->proxy;
|
2008-12-24 19:30:25 -05:00
|
|
|
else
|
2010-08-13 10:39:57 -04:00
|
|
|
proxy = wl_hash_table_lookup(display->objects, id);
|
2008-12-30 11:03:33 -05:00
|
|
|
|
2011-02-18 15:28:54 -05:00
|
|
|
if (proxy == NULL || proxy->object.implementation == NULL) {
|
2010-08-30 09:47:36 -04:00
|
|
|
wl_connection_consume(display->connection, size);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2010-12-01 17:07:41 -05:00
|
|
|
message = &proxy->object.interface->events[opcode];
|
2010-08-30 09:47:36 -04:00
|
|
|
closure = wl_connection_demarshal(display->connection,
|
|
|
|
|
size, display->objects, message);
|
|
|
|
|
|
2011-07-18 02:00:24 -04:00
|
|
|
if (closure == NULL) {
|
|
|
|
|
fprintf(stderr, "Error demarshalling event: %m\n");
|
|
|
|
|
abort();
|
|
|
|
|
}
|
|
|
|
|
|
2011-02-10 12:27:35 -05:00
|
|
|
if (wl_debug)
|
2011-07-14 18:56:40 +03:00
|
|
|
wl_closure_print(closure, &proxy->object, false);
|
2011-02-10 12:27:35 -05:00
|
|
|
|
2011-02-18 15:28:54 -05:00
|
|
|
wl_closure_invoke(closure, &proxy->object,
|
|
|
|
|
proxy->object.implementation[opcode],
|
|
|
|
|
proxy->user_data);
|
2010-08-30 09:47:36 -04:00
|
|
|
|
|
|
|
|
wl_closure_destroy(closure);
|
2008-10-07 10:10:36 -04:00
|
|
|
}
|
|
|
|
|
|
2008-11-08 15:39:41 -05:00
|
|
|
WL_EXPORT void
|
2008-10-08 13:32:07 -04:00
|
|
|
wl_display_iterate(struct wl_display *display, uint32_t mask)
|
2008-10-07 10:10:36 -04:00
|
|
|
{
|
2008-11-25 22:40:39 -05:00
|
|
|
uint32_t p[2], object, opcode, size;
|
2008-10-07 10:10:36 -04:00
|
|
|
int len;
|
|
|
|
|
|
2011-01-22 20:41:07 +01:00
|
|
|
mask &= display->mask;
|
|
|
|
|
if (mask == 0) {
|
|
|
|
|
fprintf(stderr,
|
|
|
|
|
"wl_display_iterate called with unsolicited flags");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2008-10-08 13:32:07 -04:00
|
|
|
len = wl_connection_data(display->connection, mask);
|
2008-10-07 10:10:36 -04:00
|
|
|
while (len > 0) {
|
2008-10-08 13:32:07 -04:00
|
|
|
if (len < sizeof p)
|
2008-10-07 10:10:36 -04:00
|
|
|
break;
|
|
|
|
|
|
2008-10-08 13:32:07 -04:00
|
|
|
wl_connection_copy(display->connection, p, sizeof p);
|
2008-11-25 22:40:39 -05:00
|
|
|
object = p[0];
|
2008-10-07 10:10:36 -04:00
|
|
|
opcode = p[1] & 0xffff;
|
|
|
|
|
size = p[1] >> 16;
|
2008-10-08 13:32:07 -04:00
|
|
|
if (len < size)
|
2008-10-07 10:10:36 -04:00
|
|
|
break;
|
|
|
|
|
|
2008-11-25 22:40:39 -05:00
|
|
|
handle_event(display, object, opcode, size);
|
2008-11-02 10:12:29 -05:00
|
|
|
len -= size;
|
2008-10-07 10:10:36 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (len < 0) {
|
|
|
|
|
fprintf(stderr, "read error: %m\n");
|
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-05-10 17:51:52 +01:00
|
|
|
WL_EXPORT void
|
|
|
|
|
wl_display_flush(struct wl_display *display)
|
|
|
|
|
{
|
|
|
|
|
while (display->mask & WL_DISPLAY_WRITABLE)
|
|
|
|
|
wl_display_iterate (display, WL_DISPLAY_WRITABLE);
|
|
|
|
|
}
|
|
|
|
|
|
2008-11-23 23:41:08 -05:00
|
|
|
WL_EXPORT uint32_t
|
|
|
|
|
wl_display_allocate_id(struct wl_display *display)
|
|
|
|
|
{
|
2008-12-21 22:45:33 -05:00
|
|
|
if (display->id_count == 0) {
|
|
|
|
|
display->id_count = 256;
|
|
|
|
|
display->id = display->next_range;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
display->id_count--;
|
|
|
|
|
|
2008-11-23 23:41:08 -05:00
|
|
|
return display->id++;
|
|
|
|
|
}
|
|
|
|
|
|
2011-08-19 13:40:28 -04:00
|
|
|
WL_EXPORT void
|
|
|
|
|
wl_display_bind(struct wl_display *display,
|
|
|
|
|
uint32_t id, const char *interface, uint32_t version)
|
|
|
|
|
{
|
|
|
|
|
wl_proxy_marshal(&display->proxy,
|
|
|
|
|
WL_DISPLAY_BIND, id, interface, version);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
WL_EXPORT struct wl_callback *
|
|
|
|
|
wl_display_sync(struct wl_display *display)
|
|
|
|
|
{
|
|
|
|
|
struct wl_proxy *proxy;
|
|
|
|
|
|
|
|
|
|
proxy = wl_proxy_create(&display->proxy, &wl_callback_interface);
|
|
|
|
|
|
|
|
|
|
if (!proxy)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
wl_proxy_marshal(&display->proxy, WL_DISPLAY_SYNC, proxy);
|
|
|
|
|
|
|
|
|
|
return (struct wl_callback *) proxy;
|
|
|
|
|
}
|
|
|
|
|
|
2009-09-18 09:49:21 -04:00
|
|
|
WL_EXPORT void
|
2010-08-17 21:23:10 -04:00
|
|
|
wl_proxy_set_user_data(struct wl_proxy *proxy, void *user_data)
|
2009-09-18 09:49:21 -04:00
|
|
|
{
|
2010-08-10 14:02:48 -04:00
|
|
|
proxy->user_data = user_data;
|
2009-09-18 09:49:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
WL_EXPORT void *
|
2010-08-17 21:23:10 -04:00
|
|
|
wl_proxy_get_user_data(struct wl_proxy *proxy)
|
2009-09-18 09:49:21 -04:00
|
|
|
{
|
2010-08-10 14:02:48 -04:00
|
|
|
return proxy->user_data;
|
2009-09-18 09:49:21 -04:00
|
|
|
}
|