blob: 8fcc8095bee713a529b53eaa1cf8eb4ccd0e553b [file] [log] [blame]
Kristian Høgsbergffd710e2008-12-02 15:15:01 -05001/*
2 * Copyright © 2008 Kristian Høgsberg
3 *
4 * Permission to use, copy, modify, distribute, and sell this software and its
5 * documentation for any purpose is hereby granted without fee, provided that
6 * the above copyright notice appear in all copies and that both that copyright
7 * notice and this permission notice appear in supporting documentation, and
8 * that the name of the copyright holders not be used in advertising or
9 * publicity pertaining to distribution of the software without specific,
10 * written prior permission. The copyright holders make no representations
11 * about the suitability of this software for any purpose. It is provided "as
12 * is" without express or implied warranty.
13 *
14 * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
15 * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
16 * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
17 * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
18 * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
19 * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
20 * OF THIS SOFTWARE.
21 */
22
Kristian Høgsberga67a71a2008-10-07 10:10:36 -040023#include <stdlib.h>
24#include <stdint.h>
25#include <stddef.h>
26#include <stdio.h>
27#include <errno.h>
28#include <string.h>
29#include <unistd.h>
30#include <sys/socket.h>
31#include <sys/un.h>
32#include <ctype.h>
Kristian Høgsbergfe831a72008-12-21 21:50:23 -050033#include <assert.h>
Kristian Høgsberga67a71a2008-10-07 10:10:36 -040034#include <sys/poll.h>
35
Kristian Høgsberga1f3f602010-08-03 09:26:44 -040036#include "wayland-client-protocol.h"
Kristian Høgsberg427524a2008-10-08 13:32:07 -040037#include "connection.h"
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -050038#include "wayland-util.h"
Kristian Høgsberg427524a2008-10-08 13:32:07 -040039#include "wayland-client.h"
40
Kristian Høgsberga67a71a2008-10-07 10:10:36 -040041static const char socket_name[] = "\0wayland";
42
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -050043struct wl_global {
44 uint32_t id;
45 char *interface;
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050046 uint32_t version;
Kristian Høgsberg94448c02008-12-30 11:03:33 -050047 struct wl_proxy *proxy;
48 struct wl_list link;
49};
50
51struct wl_global_listener {
52 wl_display_global_func_t handler;
53 void *data;
54 struct wl_list link;
55};
56
57struct wl_listener {
58 void (**implementation)(void);
59 void *data;
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -050060 struct wl_list link;
61};
62
Kristian Høgsberga67a71a2008-10-07 10:10:36 -040063struct wl_proxy {
Kristian Høgsbergfabd4392008-12-22 18:06:49 -050064 struct wl_object base;
Kristian Høgsbergfe831a72008-12-21 21:50:23 -050065 struct wl_display *display;
Kristian Høgsberg94448c02008-12-30 11:03:33 -050066 struct wl_list listener_list;
Kristian Høgsbergecf65fe2009-09-18 09:49:21 -040067 void *user_data;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -040068};
69
Kristian Høgsbergd2412e22008-12-15 20:35:24 -050070struct wl_compositor {
71 struct wl_proxy proxy;
72};
73
Kristian Høgsberga67a71a2008-10-07 10:10:36 -040074struct wl_surface {
75 struct wl_proxy proxy;
76};
77
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -050078struct wl_visual {
79 struct wl_proxy proxy;
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -050080};
81
Kristian Høgsberg12ea62e2008-12-22 16:37:07 -050082struct wl_output {
83 struct wl_proxy proxy;
Kristian Høgsberg12ea62e2008-12-22 16:37:07 -050084};
85
Kristian Høgsberg83fc0612010-08-04 22:44:55 -040086struct wl_shell {
87 struct wl_proxy proxy;
88};
89
Kristian Høgsberg5fcd0aa2010-08-09 14:43:33 -040090struct wl_drm {
91 struct wl_proxy proxy;
92};
93
94struct wl_buffer {
95 struct wl_proxy proxy;
96};
97
Kristian Høgsberg94448c02008-12-30 11:03:33 -050098struct wl_input_device {
99 struct wl_proxy proxy;
100};
101
102struct wl_display {
103 struct wl_proxy proxy;
104 struct wl_connection *connection;
105 int fd;
106 uint32_t id, id_count, next_range;
107 uint32_t mask;
Kristian Høgsbergf52e03f2010-02-26 11:42:59 -0500108 struct wl_hash_table *objects;
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500109 struct wl_list global_list;
110 struct wl_listener listener;
111 struct wl_list global_listener_list;
112
113 struct wl_visual *argb_visual;
114 struct wl_visual *premultiplied_argb_visual;
115 struct wl_visual *rgb_visual;
116
117 wl_display_update_func_t update;
118 void *update_data;
119
120 wl_display_global_func_t global_handler;
121 void *global_handler_data;
122
123 struct wl_compositor *compositor;
124};
125
Kristian Høgsbergfb590842008-11-07 14:27:23 -0500126static int
127connection_update(struct wl_connection *connection,
128 uint32_t mask, void *data)
129{
130 struct wl_display *display = data;
131
132 display->mask = mask;
133 if (display->update)
134 return display->update(display->mask,
135 display->update_data);
136
137 return 0;
138}
139
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500140WL_EXPORT int
141wl_object_implements(struct wl_object *object,
142 const char *interface, int version)
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500143{
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500144 return strcmp(object->interface->name, interface) == 0 &&
145 object->interface->version >= version;
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500146}
147
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500148WL_EXPORT struct wl_global_listener *
149wl_display_add_global_listener(struct wl_display *display,
150 wl_display_global_func_t handler, void *data)
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500151{
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500152 struct wl_global_listener *listener;
153 struct wl_global *global;
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500154
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500155 listener = malloc(sizeof *listener);
156 if (listener == NULL)
157 return NULL;
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500158
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500159 listener->handler = handler;
160 listener->data = data;
161 wl_list_insert(display->global_listener_list.prev, &listener->link);
162
163 /* FIXME: Need a destructor for void *data? */
164
Kristian Høgsberga5db5892010-02-26 10:28:44 -0500165 wl_list_for_each(global, &display->global_list, link)
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500166 if (global->proxy != NULL)
167 (*handler)(display, &global->proxy->base, data);
168
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500169 return listener;
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500170}
171
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500172WL_EXPORT void
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500173wl_display_remove_global_listener(struct wl_display *display,
174 struct wl_global_listener *listener)
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500175{
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500176 wl_list_remove(&listener->link);
177 free(listener);
178}
179
180static struct wl_proxy *
181wl_proxy_create_for_global(struct wl_display *display,
182 struct wl_global *global,
183 const struct wl_interface *interface)
184{
185 struct wl_proxy *proxy;
186 struct wl_global_listener *listener;
187
188 proxy = malloc(sizeof *proxy);
189 if (proxy == NULL)
190 return NULL;
191
192 proxy->base.interface = interface;
193 proxy->base.id = global->id;
194 proxy->display = display;
195 global->proxy = proxy;
196 wl_list_init(&proxy->listener_list);
Kristian Høgsbergf52e03f2010-02-26 11:42:59 -0500197 wl_hash_table_insert(display->objects, proxy->base.id, proxy);
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500198
Kristian Høgsberga5db5892010-02-26 10:28:44 -0500199 wl_list_for_each(listener, &display->global_listener_list, link)
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500200 (*listener->handler)(display, &proxy->base, listener->data);
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500201
202 return proxy;
203}
204
205static int
206wl_proxy_add_listener(struct wl_proxy *proxy, void (**implementation)(void), void *data)
207{
208 struct wl_listener *listener;
209
210 listener = malloc(sizeof *listener);
211 if (listener == NULL)
212 return -1;
213
214 listener->implementation = (void (**)(void)) implementation;
215 listener->data = data;
216 wl_list_insert(proxy->listener_list.prev, &listener->link);
217
218 return 0;
219}
220
221static void
222wl_proxy_marshal(struct wl_proxy *proxy, uint32_t opcode, ...)
223{
224 va_list ap;
225
226 va_start(ap, opcode);
227 wl_connection_vmarshal(proxy->display->connection,
228 &proxy->base, opcode, ap,
229 &proxy->base.interface->methods[opcode]);
230 va_end(ap);
231}
232
233WL_EXPORT int
234wl_output_add_listener(struct wl_output *output,
235 const struct wl_output_listener *listener,
236 void *data)
237{
238 return wl_proxy_add_listener(&output->proxy,
239 (void (**)(void)) listener, data);
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500240}
241
Kristian Høgsberg83fc0612010-08-04 22:44:55 -0400242WL_EXPORT int
243wl_shell_add_listener(struct wl_shell *shell,
244 const struct wl_shell_listener *listener,
245 void *data)
246{
247 return wl_proxy_add_listener(&shell->proxy,
248 (void (**)(void)) listener, data);
249}
250
251WL_EXPORT void
252wl_shell_move(struct wl_shell *shell,
253 struct wl_surface *surface, struct wl_input_device *device,
254 uint32_t time)
255{
256 wl_proxy_marshal(&shell->proxy,
257 WL_SHELL_MOVE, surface, device, time);
258}
259
260WL_EXPORT void
261wl_shell_resize(struct wl_shell *shell,
262 struct wl_surface *surface, struct wl_input_device *device,
263 uint32_t time, uint32_t edges)
264{
265 wl_proxy_marshal(&shell->proxy,
266 WL_SHELL_RESIZE, surface, device, time, edges);
267}
268
Kristian Høgsberg5fcd0aa2010-08-09 14:43:33 -0400269WL_EXPORT int
270wl_drm_add_listener(struct wl_drm *drm,
271 const struct wl_drm_listener *listener,
272 void *data)
273{
274 return wl_proxy_add_listener(&drm->proxy,
275 (void (**)(void)) listener, data);
276}
277
278WL_EXPORT void
279wl_drm_authenticate(struct wl_drm *drm, uint32_t id)
280{
281 wl_proxy_marshal(&drm->proxy, WL_DRM_AUTHENTICATE, id);
282}
283
284WL_EXPORT struct wl_buffer *
285wl_drm_create_buffer(struct wl_drm *drm, uint32_t name,
286 int32_t width, int32_t height,
287 uint32_t stride, struct wl_visual *visual)
288{
289 struct wl_buffer *buffer;
290
291 buffer = malloc(sizeof *buffer);
292 if (buffer == NULL)
293 return NULL;
294
295 buffer->proxy.base.interface = &wl_buffer_interface;
296 buffer->proxy.base.id = wl_display_allocate_id(drm->proxy.display);
297 buffer->proxy.display = drm->proxy.display;
298 wl_hash_table_insert(drm->proxy.display->objects,
299 drm->proxy.base.id, buffer);
300 wl_proxy_marshal(&drm->proxy, WL_DRM_CREATE_BUFFER,
301 buffer, name, width, height, stride, visual);
302
303 return buffer;
304}
305
306WL_EXPORT void
307wl_buffer_destroy(struct wl_buffer *buffer)
308{
309 wl_proxy_marshal(&buffer->proxy, WL_BUFFER_DESTROY);
310}
311
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -0500312static void
313add_visual(struct wl_display *display, struct wl_global *global)
314{
315 struct wl_visual *visual;
316
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500317 visual = (struct wl_visual *)
318 wl_proxy_create_for_global(display, global,
319 &wl_visual_interface);
320 if (display->argb_visual == NULL)
321 display->argb_visual = visual;
322 else if (display->premultiplied_argb_visual == NULL)
323 display->premultiplied_argb_visual = visual;
324 else
325 display->rgb_visual = visual;
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -0500326}
327
328WL_EXPORT struct wl_visual *
329wl_display_get_argb_visual(struct wl_display *display)
330{
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500331 return display->argb_visual;
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -0500332}
333
334WL_EXPORT struct wl_visual *
335wl_display_get_premultiplied_argb_visual(struct wl_display *display)
336{
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500337 return display->premultiplied_argb_visual;
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -0500338}
339
340WL_EXPORT struct wl_visual *
341wl_display_get_rgb_visual(struct wl_display *display)
342{
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500343 return display->rgb_visual;
344}
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -0500345
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500346WL_EXPORT int
347wl_input_device_add_listener(struct wl_input_device *input_device,
348 const struct wl_input_device_listener *listener,
349 void *data)
350{
351 return wl_proxy_add_listener(&input_device->proxy,
352 (void (**)(void)) listener, data);
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -0500353}
354
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500355static void
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500356display_handle_invalid_object(void *data,
357 struct wl_display *display, uint32_t id)
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500358{
359 fprintf(stderr, "sent request to invalid object\n");
Kristian Høgsbergc071f4d2010-08-05 17:44:31 -0400360 abort();
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500361}
362
363static void
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500364display_handle_invalid_method(void *data,
365 struct wl_display *display,
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500366 uint32_t id, uint32_t opcode)
367{
368 fprintf(stderr, "sent invalid request opcode\n");
Kristian Høgsbergc071f4d2010-08-05 17:44:31 -0400369 abort();
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500370}
371
372static void
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500373display_handle_no_memory(void *data,
374 struct wl_display *display)
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500375{
376 fprintf(stderr, "server out of memory\n");
Kristian Høgsbergc071f4d2010-08-05 17:44:31 -0400377 abort();
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500378}
379
380static void
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500381display_handle_global(void *data,
382 struct wl_display *display,
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500383 uint32_t id, const char *interface, uint32_t version)
384{
385 struct wl_global *global;
386
387 global = malloc(sizeof *global);
388 if (global == NULL)
389 return;
390
391 global->id = id;
392 global->interface = strdup(interface);
393 global->version = version;
394 wl_list_insert(display->global_list.prev, &global->link);
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500395
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500396 if (strcmp(global->interface, "display") == 0)
Kristian Høgsbergf52e03f2010-02-26 11:42:59 -0500397 wl_hash_table_insert(display->objects,
398 id, &display->proxy.base);
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500399 else if (strcmp(global->interface, "compositor") == 0)
400 display->compositor = (struct wl_compositor *)
401 wl_proxy_create_for_global(display, global,
402 &wl_compositor_interface);
403 else if (strcmp(global->interface, "visual") == 0)
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500404 add_visual(display, global);
405 else if (strcmp(global->interface, "output") == 0)
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500406 wl_proxy_create_for_global(display, global,
407 &wl_output_interface);
408 else if (strcmp(global->interface, "input_device") == 0)
409 wl_proxy_create_for_global(display, global,
410 &wl_input_device_interface);
Kristian Høgsberg83fc0612010-08-04 22:44:55 -0400411 else if (strcmp(global->interface, "shell") == 0)
412 wl_proxy_create_for_global(display, global,
413 &wl_shell_interface);
Kristian Høgsberg5fcd0aa2010-08-09 14:43:33 -0400414 else if (strcmp(global->interface, "drm") == 0)
415 wl_proxy_create_for_global(display, global,
416 &wl_drm_interface);
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500417}
418
419static void
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500420display_handle_range(void *data,
421 struct wl_display *display, uint32_t range)
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500422{
423 display->next_range = range;
424}
425
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500426static const struct wl_display_listener display_listener = {
427 display_handle_invalid_object,
428 display_handle_invalid_method,
429 display_handle_no_memory,
430 display_handle_global,
431 display_handle_range
432};
433
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500434WL_EXPORT struct wl_display *
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500435wl_display_create(const char *name, size_t name_size)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400436{
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400437 struct wl_display *display;
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500438 struct sockaddr_un addr;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400439 socklen_t size;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400440
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400441 display = malloc(sizeof *display);
442 if (display == NULL)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400443 return NULL;
444
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400445 memset(display, 0, sizeof *display);
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400446 display->fd = socket(PF_LOCAL, SOCK_STREAM, 0);
447 if (display->fd < 0) {
448 free(display);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400449 return NULL;
450 }
451
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500452 addr.sun_family = AF_LOCAL;
453 memcpy(addr.sun_path, name, name_size);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400454
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500455 size = offsetof (struct sockaddr_un, sun_path) + name_size;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400456
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500457 if (connect(display->fd, (struct sockaddr *) &addr, size) < 0) {
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400458 close(display->fd);
459 free(display);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400460 return NULL;
461 }
462
Kristian Høgsbergf52e03f2010-02-26 11:42:59 -0500463 display->objects = wl_hash_table_create();
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500464 wl_list_init(&display->global_list);
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500465 wl_list_init(&display->global_listener_list);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400466
Kristian Høgsbergfabd4392008-12-22 18:06:49 -0500467 display->proxy.base.interface = &wl_display_interface;
Kristian Høgsbergfabd4392008-12-22 18:06:49 -0500468 display->proxy.base.id = 1;
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500469 display->proxy.display = display;
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500470 wl_list_init(&display->proxy.listener_list);
471
472 display->listener.implementation = (void(**)(void)) &display_listener;
473 wl_list_insert(display->proxy.listener_list.prev, &display->listener.link);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400474
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400475 display->connection = wl_connection_create(display->fd,
Kristian Høgsbergfb590842008-11-07 14:27:23 -0500476 connection_update,
477 display);
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400478
479 return display;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400480}
481
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500482WL_EXPORT void
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400483wl_display_destroy(struct wl_display *display)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400484{
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400485 wl_connection_destroy(display->connection);
486 close(display->fd);
487 free(display);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400488}
489
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500490WL_EXPORT uint32_t
Kristian Høgsberg8049cbb2008-12-21 22:50:32 -0500491wl_display_get_object_id(struct wl_display *display,
492 const char *interface, uint32_t version)
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500493{
494 struct wl_global *global;
495
Kristian Høgsberga5db5892010-02-26 10:28:44 -0500496 wl_list_for_each(global, &display->global_list, link)
Kristian Høgsberg8049cbb2008-12-21 22:50:32 -0500497 if (strcmp(global->interface, interface) == 0 &&
498 global->version >= version)
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500499 return global->id;
500
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500501 return 0;
502}
503
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500504WL_EXPORT int
Kristian Høgsbergfb590842008-11-07 14:27:23 -0500505wl_display_get_fd(struct wl_display *display,
506 wl_display_update_func_t update, void *data)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400507{
Kristian Høgsbergfb590842008-11-07 14:27:23 -0500508 display->update = update;
509 display->update_data = data;
510
511 display->update(display->mask, display->update_data);
512
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400513 return display->fd;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400514}
515
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500516static void
Kristian Høgsberg40979232008-11-25 22:40:39 -0500517handle_event(struct wl_display *display,
Kristian Høgsbergfabd4392008-12-22 18:06:49 -0500518 uint32_t id, uint32_t opcode, uint32_t size)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400519{
Kristian Høgsbergbf967b42008-12-21 20:25:16 -0500520 uint32_t p[32];
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500521 struct wl_listener *listener;
522 struct wl_proxy *proxy;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400523
Kristian Høgsberg5a27f3e2008-11-02 10:55:25 -0500524 wl_connection_copy(display->connection, p, size);
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500525 if (id == 1)
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500526 proxy = &display->proxy;
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500527 else
Kristian Høgsbergf52e03f2010-02-26 11:42:59 -0500528 proxy = (struct wl_proxy *)
529 wl_hash_table_lookup(display->objects, id);
Kristian Høgsbergfabd4392008-12-22 18:06:49 -0500530
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500531 if (proxy != NULL) {
532 if (wl_list_empty(&proxy->listener_list)) {
533 printf("proxy found for object %d, opcode %d, but no listeners\n",
534 id, opcode);
535 }
536
Kristian Høgsberga5db5892010-02-26 10:28:44 -0500537 wl_list_for_each(listener, &proxy->listener_list, link)
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500538 wl_connection_demarshal(display->connection,
539 size,
540 display->objects,
541 listener->implementation[opcode],
542 listener->data,
543 &proxy->base,
544 &proxy->base.interface->events[opcode]);
545
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500546 }
Kristian Høgsbergb3131d92008-12-24 19:30:25 -0500547
Kristian Høgsberg5a27f3e2008-11-02 10:55:25 -0500548 wl_connection_consume(display->connection, size);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400549}
550
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500551WL_EXPORT void
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400552wl_display_iterate(struct wl_display *display, uint32_t mask)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400553{
Kristian Høgsberg40979232008-11-25 22:40:39 -0500554 uint32_t p[2], object, opcode, size;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400555 int len;
556
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400557 len = wl_connection_data(display->connection, mask);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400558 while (len > 0) {
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400559 if (len < sizeof p)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400560 break;
561
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400562 wl_connection_copy(display->connection, p, sizeof p);
Kristian Høgsberg40979232008-11-25 22:40:39 -0500563 object = p[0];
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400564 opcode = p[1] & 0xffff;
565 size = p[1] >> 16;
Kristian Høgsberg427524a2008-10-08 13:32:07 -0400566 if (len < size)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400567 break;
568
Kristian Høgsberg40979232008-11-25 22:40:39 -0500569 handle_event(display, object, opcode, size);
Kristian Høgsbergf9bc7952008-11-02 10:12:29 -0500570 len -= size;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400571 }
572
573 if (len < 0) {
574 fprintf(stderr, "read error: %m\n");
575 exit(EXIT_FAILURE);
576 }
577}
578
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500579WL_EXPORT uint32_t
580wl_display_allocate_id(struct wl_display *display)
581{
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500582 if (display->id_count == 0) {
583 display->id_count = 256;
584 display->id = display->next_range;
585 }
586
587 display->id_count--;
588
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500589 return display->id++;
590}
591
592WL_EXPORT void
593wl_display_write(struct wl_display *display, const void *data, size_t count)
594{
595 wl_connection_write(display->connection, data, count);
596}
597
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500598WL_EXPORT struct wl_compositor *
599wl_display_get_compositor(struct wl_display *display)
600{
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500601 return display->compositor;
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500602}
603
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500604WL_EXPORT int
605wl_compositor_add_listener(struct wl_compositor *compositor,
606 const struct wl_compositor_listener *listener,
607 void *data)
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500608{
Kristian Høgsberg94448c02008-12-30 11:03:33 -0500609 return wl_proxy_add_listener(&compositor->proxy,
610 (void (**)(void)) listener, data);
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500611}
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400612
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500613WL_EXPORT struct wl_surface *
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500614wl_compositor_create_surface(struct wl_compositor *compositor)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400615{
616 struct wl_surface *surface;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400617
618 surface = malloc(sizeof *surface);
619 if (surface == NULL)
620 return NULL;
621
Kristian Høgsbergfabd4392008-12-22 18:06:49 -0500622 surface->proxy.base.interface = &wl_surface_interface;
623 surface->proxy.base.id = wl_display_allocate_id(compositor->proxy.display);
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500624 surface->proxy.display = compositor->proxy.display;
Kristian Høgsbergf52e03f2010-02-26 11:42:59 -0500625 wl_hash_table_insert(compositor->proxy.display->objects,
626 surface->proxy.base.id, surface);
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500627 wl_proxy_marshal(&compositor->proxy,
628 WL_COMPOSITOR_CREATE_SURFACE, surface);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400629
630 return surface;
631}
632
Kristian Høgsberg40979232008-11-25 22:40:39 -0500633WL_EXPORT void
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500634wl_compositor_commit(struct wl_compositor *compositor, uint32_t key)
Kristian Høgsberg40979232008-11-25 22:40:39 -0500635{
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500636 wl_proxy_marshal(&compositor->proxy, WL_COMPOSITOR_COMMIT, key);
Kristian Høgsberg40979232008-11-25 22:40:39 -0500637}
638
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500639WL_EXPORT void
640wl_surface_destroy(struct wl_surface *surface)
Kristian Høgsberg05eff512008-10-07 10:10:36 -0400641{
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500642 wl_proxy_marshal(&surface->proxy, WL_SURFACE_DESTROY);
Kristian Høgsberg05eff512008-10-07 10:10:36 -0400643}
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400644
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500645WL_EXPORT void
Kristian Høgsberg5fcd0aa2010-08-09 14:43:33 -0400646wl_surface_attach(struct wl_surface *surface, struct wl_buffer *buffer)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400647{
Kristian Høgsberg5fcd0aa2010-08-09 14:43:33 -0400648 wl_proxy_marshal(&surface->proxy, WL_SURFACE_ATTACH, buffer);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400649}
Kristian Høgsberg05eff512008-10-07 10:10:36 -0400650
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500651WL_EXPORT void
652wl_surface_map(struct wl_surface *surface,
653 int32_t x, int32_t y, int32_t width, int32_t height)
Kristian Høgsberg05eff512008-10-07 10:10:36 -0400654{
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500655 wl_proxy_marshal(&surface->proxy,
656 WL_SURFACE_MAP, x, y, width, height);
Kristian Høgsberg05eff512008-10-07 10:10:36 -0400657}
Kristian Høgsberg7f77bd82008-11-07 08:39:37 -0500658
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500659WL_EXPORT void
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500660wl_surface_damage(struct wl_surface *surface,
661 int32_t x, int32_t y, int32_t width, int32_t height)
Kristian Høgsberg7f77bd82008-11-07 08:39:37 -0500662{
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500663 wl_proxy_marshal(&surface->proxy,
664 WL_SURFACE_DAMAGE, x, y, width, height);
Kristian Høgsberg7f77bd82008-11-07 08:39:37 -0500665}
Kristian Høgsbergecf65fe2009-09-18 09:49:21 -0400666
667WL_EXPORT void
668wl_surface_set_user_data(struct wl_surface *surface, void *user_data)
669{
670 surface->proxy.user_data = user_data;
671}
672
673WL_EXPORT void *
674wl_surface_get_user_data(struct wl_surface *surface)
675{
676 return surface->proxy.user_data;
677}