blob: 0fe2a8de9ec85393f53cf60b819b19bee35e23d9 [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øgsberg97f1ebe2008-09-30 09:46:10 -040023#include <stdlib.h>
24#include <stdint.h>
25#include <stddef.h>
26#include <stdio.h>
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050027#include <stdarg.h>
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040028#include <errno.h>
29#include <string.h>
30#include <unistd.h>
31#include <sys/socket.h>
32#include <sys/un.h>
Kristian Høgsberg5503bf82008-11-06 10:38:17 -050033#include <dlfcn.h>
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050034#include <assert.h>
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040035#include <ffi.h>
Kristian Høgsberg680f1c72008-10-08 12:48:46 -040036
Kristian Høgsbergfe831a72008-12-21 21:50:23 -050037#include "wayland-protocol.h"
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040038#include "wayland.h"
Kristian Høgsberg680f1c72008-10-08 12:48:46 -040039#include "connection.h"
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040040
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040041struct wl_client {
Kristian Høgsberg680f1c72008-10-08 12:48:46 -040042 struct wl_connection *connection;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040043 struct wl_event_source *source;
44 struct wl_display *display;
Kristian Høgsberg94a2e862008-10-11 21:37:55 -040045 struct wl_list object_list;
Kristian Høgsbergf9bc7952008-11-02 10:12:29 -050046 struct wl_list link;
Kristian Høgsberg97079ad2008-12-21 22:45:33 -050047 uint32_t id_count;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040048};
49
50struct wl_display {
51 struct wl_object base;
52 struct wl_event_loop *loop;
Kristian Høgsberg864c4682008-12-12 11:05:17 -050053 struct wl_hash *objects;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -040054
Kristian Høgsbergd2412e22008-12-15 20:35:24 -050055 struct wl_list pending_frame_list;
Kristian Høgsbergf9212892008-10-11 18:40:23 -040056 uint32_t client_id_range;
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -050057 uint32_t id;
Kristian Høgsbergf9bc7952008-11-02 10:12:29 -050058
Kristian Høgsberg14fcff72008-11-23 19:10:23 -050059 struct wl_list global_list;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040060};
61
Kristian Høgsberg94a2e862008-10-11 21:37:55 -040062struct wl_object_ref {
63 struct wl_object *object;
64 struct wl_list link;
65};
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -050066
Kristian Høgsbergee02ca62008-12-21 23:37:12 -050067struct wl_global {
68 struct wl_object *object;
69 wl_client_connect_func_t func;
70 struct wl_list link;
71};
72
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -040073void
Kristian Høgsberg680f1c72008-10-08 12:48:46 -040074wl_client_destroy(struct wl_client *client);
Kristian Høgsbergc5089872008-10-08 10:47:59 -040075
76static void
Kristian Høgsberg5a75c902008-12-10 13:16:50 -050077wl_client_vmarshal(struct wl_client *client, struct wl_object *sender,
78 uint32_t opcode, va_list ap)
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050079{
Kristian Høgsbergfb6d68d2008-12-21 21:54:51 -050080 const struct wl_message *event;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050081 struct wl_object *object;
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050082 uint32_t args[32], length, *p, size;
83 const char *s;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050084 int i, count;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050085
86 event = &sender->interface->events[opcode];
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050087 count = strlen(event->signature);
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050088 assert(count <= ARRAY_LENGTH(args));
89
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050090 p = &args[2];
91 for (i = 0; i < count; i++) {
92 switch (event->signature[i]) {
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050093 case 'u':
94 case 'i':
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050095 *p++ = va_arg(ap, uint32_t);
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050096 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050097 case 's':
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050098 s = va_arg(ap, const char *);
99 length = strlen(s);
100 *p++ = length;
101 memcpy(p, s, length);
102 p += DIV_ROUNDUP(length, sizeof(*p));
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500103 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500104 case 'o':
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500105 object = va_arg(ap, struct wl_object *);
Kristian Høgsbergbf967b42008-12-21 20:25:16 -0500106 *p++ = object->id;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500107 break;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500108 default:
109 assert(0);
110 break;
111 }
112 }
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500113
Kristian Høgsbergbf967b42008-12-21 20:25:16 -0500114 size = (p - args) * sizeof *p;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500115 args[0] = sender->id;
116 args[1] = opcode | (size << 16);
117 wl_connection_write(client->connection, args, size);
118}
119
120static void
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500121wl_client_marshal(struct wl_client *client, struct wl_object *sender,
122 uint32_t opcode, ...)
123{
124 va_list ap;
125
126 va_start(ap, opcode);
127 wl_client_vmarshal(client, sender, opcode, ap);
128 va_end(ap);
129}
130
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500131WL_EXPORT void
132wl_client_post_event(struct wl_client *client, struct wl_object *sender,
133 uint32_t opcode, ...)
134{
135 va_list ap;
136
137 va_start(ap, opcode);
138 wl_client_vmarshal(client, sender, opcode, ap);
139 va_end(ap);
140}
141
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500142static void
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400143wl_client_demarshal(struct wl_client *client, struct wl_object *target,
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500144 uint32_t opcode, size_t size)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400145{
Kristian Høgsbergfb6d68d2008-12-21 21:54:51 -0500146 const struct wl_message *method;
Kristian Høgsberg7f77bd82008-11-07 08:39:37 -0500147 ffi_type *types[20];
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400148 ffi_cif cif;
149 uint32_t *p, result;
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -0500150 int i, count;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400151 union {
152 uint32_t uint32;
153 const char *string;
154 void *object;
155 uint32_t new_id;
Kristian Høgsberg7f77bd82008-11-07 08:39:37 -0500156 } values[20];
157 void *args[20];
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400158 struct wl_object *object;
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400159 uint32_t data[64];
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500160 void (*func)(void);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400161
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500162 method = &target->interface->methods[opcode];
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500163 count = strlen(method->signature) + 2;
164 if (count > ARRAY_LENGTH(types)) {
165 printf("too many args (%d)\n", count);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400166 return;
167 }
168
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400169 if (sizeof data < size) {
170 printf("request too big, should malloc tmp buffer here\n");
171 return;
172 }
173
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400174 types[0] = &ffi_type_pointer;
175 values[0].object = client;
176 args[0] = &values[0];
177
178 types[1] = &ffi_type_pointer;
179 values[1].object = target;
180 args[1] = &values[1];
181
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400182 wl_connection_copy(client->connection, data, size);
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400183 p = &data[2];
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500184 for (i = 2; i < count; i++) {
185 switch (method->signature[i - 2]) {
186 case 'u':
187 case 'i':
188 types[i] = &ffi_type_uint32;
189 values[i].uint32 = *p;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400190 p++;
191 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500192 case 's':
193 types[i] = &ffi_type_pointer;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400194 /* FIXME */
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500195 values[i].uint32 = *p++;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400196 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500197 case 'o':
198 types[i] = &ffi_type_pointer;
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500199 object = wl_hash_lookup(client->display->objects, *p);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400200 if (object == NULL)
201 printf("unknown object (%d)\n", *p);
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500202 values[i].object = object;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400203 p++;
204 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500205 case 'n':
206 types[i] = &ffi_type_uint32;
207 values[i].new_id = *p;
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500208 object = wl_hash_lookup(client->display->objects, *p);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400209 if (object != NULL)
210 printf("object already exists (%d)\n", *p);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400211 p++;
212 break;
213 default:
214 printf("unknown type\n");
215 break;
216 }
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500217 args[i] = &values[i];
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400218 }
219
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500220 func = target->implementation[opcode];
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500221 ffi_prep_cif(&cif, FFI_DEFAULT_ABI, count, &ffi_type_uint32, types);
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500222 ffi_call(&cif, func, &result, args);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400223}
224
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400225static void
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400226wl_client_connection_data(int fd, uint32_t mask, void *data)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400227{
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400228 struct wl_client *client = data;
229 struct wl_connection *connection = client->connection;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400230 struct wl_object *object;
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400231 uint32_t p[2], opcode, size;
232 uint32_t cmask = 0;
233 int len;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400234
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400235 if (mask & WL_EVENT_READABLE)
236 cmask |= WL_CONNECTION_READABLE;
237 if (mask & WL_EVENT_WRITEABLE)
238 cmask |= WL_CONNECTION_WRITABLE;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400239
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400240 len = wl_connection_data(connection, cmask);
241 if (len < 0) {
242 wl_client_destroy(client);
243 return;
244 }
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400245
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500246 while (len >= sizeof p) {
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400247 wl_connection_copy(connection, p, sizeof p);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400248 opcode = p[1] & 0xffff;
249 size = p[1] >> 16;
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400250 if (len < size)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400251 break;
252
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500253 object = wl_hash_lookup(client->display->objects, p[0]);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400254 if (object == NULL) {
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500255 wl_client_marshal(client, &client->display->base,
256 WL_DISPLAY_INVALID_OBJECT, p[0]);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400257 wl_connection_consume(connection, size);
258 len -= size;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400259 continue;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400260 }
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400261
262 if (opcode >= object->interface->method_count) {
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500263 wl_client_marshal(client, &client->display->base,
264 WL_DISPLAY_INVALID_METHOD, p[0], opcode);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400265 wl_connection_consume(connection, size);
266 len -= size;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400267 continue;
268 }
269
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500270 wl_client_demarshal(client, object, opcode, size);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400271 wl_connection_consume(connection, size);
272 len -= size;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400273 }
274}
275
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400276static int
277wl_client_connection_update(struct wl_connection *connection,
278 uint32_t mask, void *data)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400279{
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400280 struct wl_client *client = data;
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400281 uint32_t emask = 0;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400282
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400283 if (mask & WL_CONNECTION_READABLE)
284 emask |= WL_EVENT_READABLE;
285 if (mask & WL_CONNECTION_WRITABLE)
286 emask |= WL_EVENT_WRITEABLE;
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400287
Kristian Høgsberg4a298902008-11-28 18:35:25 -0500288 return wl_event_source_fd_update(client->source, mask);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400289}
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400290
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500291static void
292wl_display_post_range(struct wl_display *display, struct wl_client *client)
293{
294 wl_client_marshal(client, &client->display->base,
295 WL_DISPLAY_RANGE, display->client_id_range);
296 display->client_id_range += 256;
297 client->id_count += 256;
298}
299
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500300static struct wl_client *
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400301wl_client_create(struct wl_display *display, int fd)
302{
303 struct wl_client *client;
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500304 struct wl_global *global;
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400305
306 client = malloc(sizeof *client);
307 if (client == NULL)
308 return NULL;
309
310 memset(client, 0, sizeof *client);
311 client->display = display;
312 client->source = wl_event_loop_add_fd(display->loop, fd,
313 WL_EVENT_READABLE,
314 wl_client_connection_data, client);
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500315 client->connection =
316 wl_connection_create(fd, wl_client_connection_update, client);
317
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400318 wl_list_init(&client->object_list);
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500319 wl_list_init(&client->link);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400320
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500321 wl_display_post_range(display, client);
Kristian Høgsbergf9212892008-10-11 18:40:23 -0400322
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500323 global = container_of(display->global_list.next,
324 struct wl_global, link);
325 while (&global->link != &display->global_list) {
Kristian Høgsbergbf967b42008-12-21 20:25:16 -0500326 wl_client_marshal(client, &client->display->base,
327 WL_DISPLAY_GLOBAL,
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500328 global->object,
329 global->object->interface->name,
330 global->object->interface->version);
331 global = container_of(global->link.next,
332 struct wl_global, link);
333 }
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500334
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500335 global = container_of(display->global_list.next,
336 struct wl_global, link);
337 while (&global->link != &display->global_list) {
338 if (global->func)
339 global->func(client, global->object);
340 global = container_of(global->link.next,
341 struct wl_global, link);
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500342 }
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400343
344 return client;
345}
346
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500347static void
348wl_object_destroy(struct wl_object *object)
349{
350 const struct wl_surface_interface *interface =
351 (const struct wl_surface_interface *) object->implementation;
352
353 /* FIXME: Need generic object destructor. */
354 interface->destroy(NULL, (struct wl_surface *) object);
355}
356
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400357void
358wl_client_destroy(struct wl_client *client)
359{
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400360 struct wl_object_ref *ref;
361
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400362 printf("disconnect from client %p\n", client);
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400363
Kristian Høgsbergf9bc7952008-11-02 10:12:29 -0500364 wl_list_remove(&client->link);
365
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400366 while (client->object_list.next != &client->object_list) {
367 ref = container_of(client->object_list.next,
368 struct wl_object_ref, link);
369 wl_list_remove(&ref->link);
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500370 wl_object_destroy(ref->object);
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400371 free(ref);
372 }
373
Kristian Høgsberg4a298902008-11-28 18:35:25 -0500374 wl_event_source_remove(client->source);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400375 wl_connection_destroy(client->connection);
376 free(client);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400377}
378
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500379WL_EXPORT int
380wl_client_add_surface(struct wl_client *client,
381 struct wl_surface *surface,
382 const struct wl_surface_interface *implementation,
383 uint32_t id)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400384{
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500385 struct wl_display *display = client->display;
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400386 struct wl_object_ref *ref;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400387
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500388 if (client->id_count-- < 64)
389 wl_display_post_range(display, client);
390
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500391 surface->base.id = id;
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500392 surface->base.interface = &wl_surface_interface;
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500393 surface->base.implementation = (void (**)(void)) implementation;
394 surface->client = client;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400395
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400396 ref = malloc(sizeof *ref);
397 if (ref == NULL) {
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500398 wl_client_marshal(client, &display->base,
399 WL_DISPLAY_NO_MEMORY);
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400400 return -1;
401 }
402
403 ref->object = &surface->base;
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500404 wl_hash_insert(display->objects, &surface->base);
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400405 wl_list_insert(client->object_list.prev, &ref->link);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400406
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400407 return 0;
408}
409
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500410WL_EXPORT void
411wl_client_send_acknowledge(struct wl_client *client,
412 struct wl_compositor *compositor,
413 uint32_t key, uint32_t frame)
Kristian Høgsberg40979232008-11-25 22:40:39 -0500414{
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500415 wl_list_remove(&client->link);
416 wl_list_insert(client->display->pending_frame_list.prev,
417 &client->link);
418 wl_client_marshal(client, &compositor->base,
419 WL_COMPOSITOR_ACKNOWLEDGE, key, frame);
420}
Kristian Høgsbergfbdbbdc2008-11-28 17:06:06 -0500421
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500422WL_EXPORT int
423wl_display_set_compositor(struct wl_display *display,
424 struct wl_compositor *compositor,
425 const struct wl_compositor_interface *implementation)
426{
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500427 compositor->base.interface = &wl_compositor_interface;
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500428 compositor->base.implementation = (void (**)(void)) implementation;
429
430 wl_display_add_object(display, &compositor->base);
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500431 if (wl_display_add_global(display, &compositor->base, NULL))
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500432 return -1;
Kristian Høgsberg40979232008-11-25 22:40:39 -0500433
434 return 0;
435}
436
Kristian Høgsberg122912c2008-12-05 11:13:50 -0500437WL_EXPORT struct wl_display *
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400438wl_display_create(void)
439{
440 struct wl_display *display;
441
442 display = malloc(sizeof *display);
443 if (display == NULL)
444 return NULL;
445
446 display->loop = wl_event_loop_create();
447 if (display->loop == NULL) {
448 free(display);
449 return NULL;
450 }
451
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500452 display->objects = wl_hash_create();
453 if (display->objects == NULL) {
454 free(display);
455 return NULL;
456 }
457
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500458 wl_list_init(&display->pending_frame_list);
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500459 wl_list_init(&display->global_list);
Kristian Høgsbergf9bc7952008-11-02 10:12:29 -0500460
Kristian Høgsbergf9212892008-10-11 18:40:23 -0400461 display->client_id_range = 256; /* Gah, arbitrary... */
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400462
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500463 display->id = 1;
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500464 display->base.interface = &wl_display_interface;
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500465 display->base.implementation = NULL;
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500466 wl_display_add_object(display, &display->base);
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500467 if (wl_display_add_global(display, &display->base, NULL)) {
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500468 wl_event_loop_destroy(display->loop);
469 free(display);
470 return NULL;
471 }
472
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400473 return display;
474}
475
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500476WL_EXPORT void
477wl_display_add_object(struct wl_display *display, struct wl_object *object)
478{
479 object->id = display->id++;
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500480 wl_hash_insert(display->objects, object);
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500481}
482
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500483WL_EXPORT int
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500484wl_display_add_global(struct wl_display *display,
485 struct wl_object *object, wl_client_connect_func_t func)
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500486{
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500487 struct wl_global *global;
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500488
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500489 global = malloc(sizeof *global);
490 if (global == NULL)
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500491 return -1;
492
Kristian Høgsbergee02ca62008-12-21 23:37:12 -0500493 global->object = object;
494 global->func = func;
495 wl_list_insert(display->global_list.prev, &global->link);
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500496
497 return 0;
498}
499
Kristian Høgsbergcddc0ad2008-11-24 00:31:49 -0500500WL_EXPORT void
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500501wl_surface_post_event(struct wl_surface *surface,
502 struct wl_object *sender,
503 uint32_t event, ...)
Kristian Høgsberg715a0812008-12-10 10:42:04 -0500504{
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500505 va_list ap;
Kristian Høgsberg715a0812008-12-10 10:42:04 -0500506
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500507 va_start(ap, event);
508 wl_client_vmarshal(surface->client, sender, event, ap);
509 va_end(ap);
Kristian Høgsberg715a0812008-12-10 10:42:04 -0500510}
511
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500512WL_EXPORT void
Kristian Høgsbergfbdbbdc2008-11-28 17:06:06 -0500513wl_display_post_frame(struct wl_display *display,
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500514 struct wl_compositor *compositor,
Kristian Høgsbergfbdbbdc2008-11-28 17:06:06 -0500515 uint32_t frame, uint32_t msecs)
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500516{
517 struct wl_client *client;
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500518
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500519 client = container_of(display->pending_frame_list.next,
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500520 struct wl_client, link);
521
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500522 while (&client->link != &display->pending_frame_list) {
523 wl_client_marshal(client, &compositor->base,
524 WL_COMPOSITOR_FRAME, frame, msecs);
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500525 client = container_of(client->link.next,
526 struct wl_client, link);
527 }
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500528
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500529 wl_list_remove(&display->pending_frame_list);
530 wl_list_init(&display->pending_frame_list);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400531}
532
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500533WL_EXPORT struct wl_event_loop *
Kristian Høgsberg5ebb3172008-10-11 19:21:35 -0400534wl_display_get_event_loop(struct wl_display *display)
535{
536 return display->loop;
537}
538
Kristian Høgsberg122912c2008-12-05 11:13:50 -0500539WL_EXPORT void
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400540wl_display_run(struct wl_display *display)
541{
542 while (1)
543 wl_event_loop_wait(display->loop);
544}
545
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400546static void
547socket_data(int fd, uint32_t mask, void *data)
548{
549 struct wl_display *display = data;
550 struct sockaddr_un name;
551 socklen_t length;
552 int client_fd;
553
554 length = sizeof name;
555 client_fd = accept (fd, (struct sockaddr *) &name, &length);
556 if (client_fd < 0)
557 fprintf(stderr, "failed to accept\n");
558
559 wl_client_create(display, client_fd);
560}
561
Kristian Høgsberg122912c2008-12-05 11:13:50 -0500562WL_EXPORT int
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500563wl_display_add_socket(struct wl_display *display,
564 const char *name, size_t name_size)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400565{
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500566 struct sockaddr_un addr;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400567 int sock;
568 socklen_t size;
569
570 sock = socket(PF_LOCAL, SOCK_STREAM, 0);
571 if (sock < 0)
572 return -1;
573
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500574 addr.sun_family = AF_LOCAL;
575 memcpy(addr.sun_path, name, name_size);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400576
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500577 size = offsetof (struct sockaddr_un, sun_path) + name_size;
578 if (bind(sock, (struct sockaddr *) &addr, size) < 0)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400579 return -1;
580
581 if (listen(sock, 1) < 0)
582 return -1;
583
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400584 wl_event_loop_add_fd(display->loop, sock,
585 WL_EVENT_READABLE,
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400586 socket_data, display);
587
588 return 0;
589}