/* * 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. */ #include #include #include #include #include #include #include #include #include #include #include #include #include "wayland-protocol.h" #include "connection.h" #include "wayland-util.h" #include "wayland-client.h" static const char socket_name[] = "\0wayland"; struct wl_global { uint32_t id; char *interface; uint32_t version; struct wl_list link; }; struct wl_proxy { const struct wl_interface *interface; uint32_t id; struct wl_display *display; }; struct wl_display { struct wl_proxy proxy; struct wl_connection *connection; int fd; uint32_t id, id_count, next_range; uint32_t mask; struct wl_list global_list; struct wl_list visual_list; wl_display_update_func_t update; void *update_data; wl_display_event_func_t event_handler; void *event_handler_data; }; struct wl_compositor { struct wl_proxy proxy; }; struct wl_surface { struct wl_proxy proxy; }; struct wl_visual { struct wl_proxy proxy; struct wl_list link; }; 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; } static void add_visual(struct wl_display *display, struct wl_global *global) { struct wl_visual *visual; visual = malloc(sizeof *visual); if (visual == NULL) return; visual->proxy.display = display; visual->proxy.id = global->id; wl_list_insert(display->visual_list.prev, &visual->link); } WL_EXPORT struct wl_visual * wl_display_get_argb_visual(struct wl_display *display) { return container_of(display->visual_list.next, struct wl_visual, link); } WL_EXPORT struct wl_visual * wl_display_get_premultiplied_argb_visual(struct wl_display *display) { return container_of(display->visual_list.next->next, struct wl_visual, link); } WL_EXPORT struct wl_visual * wl_display_get_rgb_visual(struct wl_display *display) { /* FIXME: Where's cddar when you need it... */ return container_of(display->visual_list.next->next->next, struct wl_visual, link); } WL_EXPORT struct wl_display * wl_display_create(const char *name, size_t name_size) { struct wl_display *display; struct sockaddr_un addr; socklen_t size; display = malloc(sizeof *display); if (display == NULL) return NULL; memset(display, 0, sizeof *display); display->fd = socket(PF_LOCAL, SOCK_STREAM, 0); if (display->fd < 0) { free(display); return NULL; } addr.sun_family = AF_LOCAL; memcpy(addr.sun_path, name, name_size); size = offsetof (struct sockaddr_un, sun_path) + name_size; if (connect(display->fd, (struct sockaddr *) &addr, size) < 0) { close(display->fd); free(display); return NULL; } wl_list_init(&display->global_list); wl_list_init(&display->visual_list); display->proxy.interface = &wl_display_interface; display->proxy.id = wl_display_get_object_id(display, "display", 1); display->proxy.display = display; display->connection = wl_connection_create(display->fd, connection_update, display); /* Process connection events. */ wl_display_iterate(display, WL_CONNECTION_READABLE); return display; } WL_EXPORT void wl_display_destroy(struct wl_display *display) { wl_connection_destroy(display->connection); close(display->fd); free(display); } WL_EXPORT uint32_t wl_display_get_object_id(struct wl_display *display, const char *interface, uint32_t version) { struct wl_global *global; global = container_of(display->global_list.next, struct wl_global, link); while (&global->link != &display->global_list) { if (strcmp(global->interface, interface) == 0 && global->version >= version) return global->id; global = container_of(global->link.next, struct wl_global, link); } return 0; } WL_EXPORT int wl_display_get_fd(struct wl_display *display, wl_display_update_func_t update, void *data) { display->update = update; display->update_data = data; display->update(display->mask, display->update_data); return display->fd; } static void handle_display_event(struct wl_display *display, uint32_t opcode, uint32_t *p, uint32_t size) { struct wl_global *global; uint32_t length; switch (opcode) { case WL_DISPLAY_INVALID_OBJECT: fprintf(stderr, "sent request to invalid object\n"); break; case WL_DISPLAY_INVALID_METHOD: fprintf(stderr, "sent invalid request opcode\n"); break; case WL_DISPLAY_NO_MEMORY: fprintf(stderr, "server out of memory\n"); break; case WL_DISPLAY_GLOBAL: global = malloc(sizeof *global); if (global == NULL) return; global->id = p[0]; length = p[1]; global->interface = malloc(length + 1); if (global->interface == NULL) { free(global); return; } memcpy(global->interface, &p[2], length); global->interface[length] = '\0'; global->version = p[2 + DIV_ROUNDUP(length, sizeof *p)]; wl_list_insert(display->global_list.prev, &global->link); if (strcmp(global->interface, "visual") == 0) add_visual(display, global); break; case WL_DISPLAY_RANGE: display->next_range = p[0]; break; } } static void handle_event(struct wl_display *display, uint32_t object, uint32_t opcode, uint32_t size) { uint32_t p[32]; wl_connection_copy(display->connection, p, size); if (object == 1) { handle_display_event(display, opcode, p + 2, size); } else if (display->event_handler != NULL) display->event_handler(display, object, opcode, size, p + 2, display->event_handler_data); wl_connection_consume(display->connection, size); } WL_EXPORT void wl_display_iterate(struct wl_display *display, uint32_t mask) { uint32_t p[2], object, opcode, size; int len; len = wl_connection_data(display->connection, mask); while (len > 0) { if (len < sizeof p) break; wl_connection_copy(display->connection, p, sizeof p); object = p[0]; opcode = p[1] & 0xffff; size = p[1] >> 16; if (len < size) break; handle_event(display, object, opcode, size); len -= size; } if (len < 0) { fprintf(stderr, "read error: %m\n"); exit(EXIT_FAILURE); } } WL_EXPORT void wl_display_set_event_handler(struct wl_display *display, wl_display_event_func_t handler, void *data) { /* FIXME: This needs something more generic... */ display->event_handler = handler; display->event_handler_data = data; } WL_EXPORT uint32_t wl_display_allocate_id(struct wl_display *display) { if (display->id_count == 0) { display->id_count = 256; display->id = display->next_range; } display->id_count--; return display->id++; } WL_EXPORT void wl_display_write(struct wl_display *display, const void *data, size_t count) { wl_connection_write(display->connection, data, count); } WL_EXPORT struct wl_compositor * wl_display_get_compositor(struct wl_display *display) { struct wl_compositor *compositor; uint32_t id; id = wl_display_get_object_id(display, "compositor", 1); if (id == 0) return NULL; compositor = malloc(sizeof *compositor); compositor->proxy.interface = &wl_compositor_interface; compositor->proxy.id = id; compositor->proxy.display = display; return compositor; } static void wl_proxy_vmarshal(struct wl_proxy *target, uint32_t opcode, va_list ap) { struct wl_proxy *proxy; uint32_t args[32], length, *p, size; const char *s, *signature; int i, count; signature = target->interface->methods[opcode].signature; count = strlen(signature); /* FIXME: Make sure we don't overwrite args array. */ p = &args[2]; for (i = 0; i < count; i++) { switch (signature[i]) { case 'u': case 'i': *p++ = va_arg(ap, uint32_t); break; case 's': s = va_arg(ap, const char *); length = strlen(s); *p++ = length; memcpy(p, s, length); p += DIV_ROUNDUP(length, sizeof(*p)); break; case 'n': case 'o': proxy = va_arg(ap, struct wl_proxy *); *p++ = proxy->id; break; default: assert(0); break; } } size = (p - args) * sizeof *p; args[0] = target->id; args[1] = opcode | (size << 16); wl_connection_write(target->display->connection, args, size); } static void wl_proxy_marshal(struct wl_proxy *proxy, uint32_t opcode, ...) { va_list ap; va_start(ap, opcode); wl_proxy_vmarshal(proxy, opcode, ap); va_end(ap); } WL_EXPORT struct wl_surface * wl_compositor_create_surface(struct wl_compositor *compositor) { struct wl_surface *surface; surface = malloc(sizeof *surface); if (surface == NULL) return NULL; surface->proxy.interface = &wl_surface_interface; surface->proxy.id = wl_display_allocate_id(compositor->proxy.display); surface->proxy.display = compositor->proxy.display; wl_proxy_marshal(&compositor->proxy, WL_COMPOSITOR_CREATE_SURFACE, surface); return surface; } WL_EXPORT void wl_compositor_commit(struct wl_compositor *compositor, uint32_t key) { wl_proxy_marshal(&compositor->proxy, WL_COMPOSITOR_COMMIT, key); } WL_EXPORT void wl_surface_destroy(struct wl_surface *surface) { wl_proxy_marshal(&surface->proxy, WL_SURFACE_DESTROY); } WL_EXPORT void wl_surface_attach(struct wl_surface *surface, uint32_t name, int32_t width, int32_t height, uint32_t stride, struct wl_visual *visual) { wl_proxy_marshal(&surface->proxy, WL_SURFACE_ATTACH, name, width, height, stride, visual); } WL_EXPORT void wl_surface_map(struct wl_surface *surface, int32_t x, int32_t y, int32_t width, int32_t height) { wl_proxy_marshal(&surface->proxy, WL_SURFACE_MAP, x, y, width, height); } WL_EXPORT void wl_surface_copy(struct wl_surface *surface, int32_t dst_x, int32_t dst_y, uint32_t name, uint32_t stride, int32_t x, int32_t y, int32_t width, int32_t height) { wl_proxy_marshal(&surface->proxy, WL_SURFACE_COPY, dst_x, dst_y, name, stride, x, y, width, height); } WL_EXPORT void wl_surface_damage(struct wl_surface *surface, int32_t x, int32_t y, int32_t width, int32_t height) { wl_proxy_marshal(&surface->proxy, WL_SURFACE_DAMAGE, x, y, width, height); }