blob: 6ad18876e9f4968084309ca760f7fb6845529601 [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øgsberg97f1ebe2008-09-30 09:46:10 -040067void
Kristian Høgsberg680f1c72008-10-08 12:48:46 -040068wl_client_destroy(struct wl_client *client);
Kristian Høgsbergc5089872008-10-08 10:47:59 -040069
70static void
Kristian Høgsberg5a75c902008-12-10 13:16:50 -050071wl_client_vmarshal(struct wl_client *client, struct wl_object *sender,
72 uint32_t opcode, va_list ap)
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050073{
Kristian Høgsbergfb6d68d2008-12-21 21:54:51 -050074 const struct wl_message *event;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050075 struct wl_object *object;
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050076 uint32_t args[32], length, *p, size;
77 const char *s;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050078 int i, count;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050079
80 event = &sender->interface->events[opcode];
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050081 count = strlen(event->signature);
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050082 assert(count <= ARRAY_LENGTH(args));
83
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050084 p = &args[2];
85 for (i = 0; i < count; i++) {
86 switch (event->signature[i]) {
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050087 case 'u':
88 case 'i':
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050089 *p++ = va_arg(ap, uint32_t);
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050090 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050091 case 's':
Kristian Høgsbergbf967b42008-12-21 20:25:16 -050092 s = va_arg(ap, const char *);
93 length = strlen(s);
94 *p++ = length;
95 memcpy(p, s, length);
96 p += DIV_ROUNDUP(length, sizeof(*p));
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050097 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -050098 case 'o':
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -050099 object = va_arg(ap, struct wl_object *);
Kristian Høgsbergbf967b42008-12-21 20:25:16 -0500100 *p++ = object->id;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500101 break;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500102 default:
103 assert(0);
104 break;
105 }
106 }
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500107
Kristian Høgsbergbf967b42008-12-21 20:25:16 -0500108 size = (p - args) * sizeof *p;
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500109 args[0] = sender->id;
110 args[1] = opcode | (size << 16);
111 wl_connection_write(client->connection, args, size);
112}
113
114static void
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500115wl_client_marshal(struct wl_client *client, struct wl_object *sender,
116 uint32_t opcode, ...)
117{
118 va_list ap;
119
120 va_start(ap, opcode);
121 wl_client_vmarshal(client, sender, opcode, ap);
122 va_end(ap);
123}
124
125static void
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400126wl_client_demarshal(struct wl_client *client, struct wl_object *target,
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500127 uint32_t opcode, size_t size)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400128{
Kristian Høgsbergfb6d68d2008-12-21 21:54:51 -0500129 const struct wl_message *method;
Kristian Høgsberg7f77bd82008-11-07 08:39:37 -0500130 ffi_type *types[20];
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400131 ffi_cif cif;
132 uint32_t *p, result;
Kristian Høgsbergde31d5c2008-12-18 17:55:33 -0500133 int i, count;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400134 union {
135 uint32_t uint32;
136 const char *string;
137 void *object;
138 uint32_t new_id;
Kristian Høgsberg7f77bd82008-11-07 08:39:37 -0500139 } values[20];
140 void *args[20];
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400141 struct wl_object *object;
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400142 uint32_t data[64];
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500143 void (*func)(void);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400144
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500145 method = &target->interface->methods[opcode];
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500146 count = strlen(method->signature) + 2;
147 if (count > ARRAY_LENGTH(types)) {
148 printf("too many args (%d)\n", count);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400149 return;
150 }
151
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400152 if (sizeof data < size) {
153 printf("request too big, should malloc tmp buffer here\n");
154 return;
155 }
156
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400157 types[0] = &ffi_type_pointer;
158 values[0].object = client;
159 args[0] = &values[0];
160
161 types[1] = &ffi_type_pointer;
162 values[1].object = target;
163 args[1] = &values[1];
164
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400165 wl_connection_copy(client->connection, data, size);
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400166 p = &data[2];
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500167 for (i = 2; i < count; i++) {
168 switch (method->signature[i - 2]) {
169 case 'u':
170 case 'i':
171 types[i] = &ffi_type_uint32;
172 values[i].uint32 = *p;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400173 p++;
174 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500175 case 's':
176 types[i] = &ffi_type_pointer;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400177 /* FIXME */
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500178 values[i].uint32 = *p++;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400179 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500180 case 'o':
181 types[i] = &ffi_type_pointer;
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500182 object = wl_hash_lookup(client->display->objects, *p);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400183 if (object == NULL)
184 printf("unknown object (%d)\n", *p);
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500185 values[i].object = object;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400186 p++;
187 break;
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500188 case 'n':
189 types[i] = &ffi_type_uint32;
190 values[i].new_id = *p;
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500191 object = wl_hash_lookup(client->display->objects, *p);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400192 if (object != NULL)
193 printf("object already exists (%d)\n", *p);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400194 p++;
195 break;
196 default:
197 printf("unknown type\n");
198 break;
199 }
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500200 args[i] = &values[i];
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400201 }
202
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500203 func = target->implementation[opcode];
Kristian Høgsberg87e0a382008-12-05 10:38:42 -0500204 ffi_prep_cif(&cif, FFI_DEFAULT_ABI, count, &ffi_type_uint32, types);
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500205 ffi_call(&cif, func, &result, args);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400206}
207
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400208static void
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400209wl_client_connection_data(int fd, uint32_t mask, void *data)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400210{
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400211 struct wl_client *client = data;
212 struct wl_connection *connection = client->connection;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400213 struct wl_object *object;
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400214 uint32_t p[2], opcode, size;
215 uint32_t cmask = 0;
216 int len;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400217
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400218 if (mask & WL_EVENT_READABLE)
219 cmask |= WL_CONNECTION_READABLE;
220 if (mask & WL_EVENT_WRITEABLE)
221 cmask |= WL_CONNECTION_WRITABLE;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400222
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400223 len = wl_connection_data(connection, cmask);
224 if (len < 0) {
225 wl_client_destroy(client);
226 return;
227 }
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400228
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500229 while (len >= sizeof p) {
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400230 wl_connection_copy(connection, p, sizeof p);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400231 opcode = p[1] & 0xffff;
232 size = p[1] >> 16;
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400233 if (len < size)
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400234 break;
235
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500236 object = wl_hash_lookup(client->display->objects, p[0]);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400237 if (object == NULL) {
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500238 wl_client_marshal(client, &client->display->base,
239 WL_DISPLAY_INVALID_OBJECT, p[0]);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400240 wl_connection_consume(connection, size);
241 len -= size;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400242 continue;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400243 }
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400244
245 if (opcode >= object->interface->method_count) {
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500246 wl_client_marshal(client, &client->display->base,
247 WL_DISPLAY_INVALID_METHOD, p[0], opcode);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400248 wl_connection_consume(connection, size);
249 len -= size;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400250 continue;
251 }
252
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500253 wl_client_demarshal(client, object, opcode, size);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400254 wl_connection_consume(connection, size);
255 len -= size;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400256 }
257}
258
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400259static int
260wl_client_connection_update(struct wl_connection *connection,
261 uint32_t mask, void *data)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400262{
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400263 struct wl_client *client = data;
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400264 uint32_t emask = 0;
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400265
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400266 if (mask & WL_CONNECTION_READABLE)
267 emask |= WL_EVENT_READABLE;
268 if (mask & WL_CONNECTION_WRITABLE)
269 emask |= WL_EVENT_WRITEABLE;
Kristian Høgsbergc5089872008-10-08 10:47:59 -0400270
Kristian Høgsberg4a298902008-11-28 18:35:25 -0500271 return wl_event_source_fd_update(client->source, mask);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400272}
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400273
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500274static void
275wl_display_post_range(struct wl_display *display, struct wl_client *client)
276{
277 wl_client_marshal(client, &client->display->base,
278 WL_DISPLAY_RANGE, display->client_id_range);
279 display->client_id_range += 256;
280 client->id_count += 256;
281}
282
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500283static struct wl_client *
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400284wl_client_create(struct wl_display *display, int fd)
285{
286 struct wl_client *client;
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500287 struct wl_object_ref *ref;
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400288
289 client = malloc(sizeof *client);
290 if (client == NULL)
291 return NULL;
292
293 memset(client, 0, sizeof *client);
294 client->display = display;
295 client->source = wl_event_loop_add_fd(display->loop, fd,
296 WL_EVENT_READABLE,
297 wl_client_connection_data, client);
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500298 client->connection =
299 wl_connection_create(fd, wl_client_connection_update, client);
300
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400301 wl_list_init(&client->object_list);
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500302 wl_list_init(&client->link);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400303
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500304 wl_display_post_range(display, client);
Kristian Høgsbergf9212892008-10-11 18:40:23 -0400305
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500306 ref = container_of(display->global_list.next,
307 struct wl_object_ref, link);
308 while (&ref->link != &display->global_list) {
Kristian Høgsbergbf967b42008-12-21 20:25:16 -0500309 wl_client_marshal(client, &client->display->base,
310 WL_DISPLAY_GLOBAL,
311 ref->object,
312 ref->object->interface->name,
313 ref->object->interface->version);
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500314
315 ref = container_of(ref->link.next,
316 struct wl_object_ref, link);
317 }
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400318
319 return client;
320}
321
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500322static void
323wl_object_destroy(struct wl_object *object)
324{
325 const struct wl_surface_interface *interface =
326 (const struct wl_surface_interface *) object->implementation;
327
328 /* FIXME: Need generic object destructor. */
329 interface->destroy(NULL, (struct wl_surface *) object);
330}
331
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400332void
333wl_client_destroy(struct wl_client *client)
334{
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400335 struct wl_object_ref *ref;
336
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400337 printf("disconnect from client %p\n", client);
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400338
Kristian Høgsbergf9bc7952008-11-02 10:12:29 -0500339 wl_list_remove(&client->link);
340
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400341 while (client->object_list.next != &client->object_list) {
342 ref = container_of(client->object_list.next,
343 struct wl_object_ref, link);
344 wl_list_remove(&ref->link);
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500345 wl_object_destroy(ref->object);
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400346 free(ref);
347 }
348
Kristian Høgsberg4a298902008-11-28 18:35:25 -0500349 wl_event_source_remove(client->source);
Kristian Høgsberg680f1c72008-10-08 12:48:46 -0400350 wl_connection_destroy(client->connection);
351 free(client);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400352}
353
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500354WL_EXPORT int
355wl_client_add_surface(struct wl_client *client,
356 struct wl_surface *surface,
357 const struct wl_surface_interface *implementation,
358 uint32_t id)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400359{
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500360 struct wl_display *display = client->display;
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400361 struct wl_object_ref *ref;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400362
Kristian Høgsberg97079ad2008-12-21 22:45:33 -0500363 if (client->id_count-- < 64)
364 wl_display_post_range(display, client);
365
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500366 surface->base.id = id;
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500367 surface->base.interface = &wl_surface_interface;
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500368 surface->base.implementation = (void (**)(void)) implementation;
369 surface->client = client;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400370
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400371 ref = malloc(sizeof *ref);
372 if (ref == NULL) {
Kristian Høgsbergc2b633e2008-11-28 19:12:45 -0500373 wl_client_marshal(client, &display->base,
374 WL_DISPLAY_NO_MEMORY);
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400375 return -1;
376 }
377
378 ref->object = &surface->base;
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500379 wl_hash_insert(display->objects, &surface->base);
Kristian Høgsberg94a2e862008-10-11 21:37:55 -0400380 wl_list_insert(client->object_list.prev, &ref->link);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400381
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400382 return 0;
383}
384
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500385WL_EXPORT void
386wl_client_send_acknowledge(struct wl_client *client,
387 struct wl_compositor *compositor,
388 uint32_t key, uint32_t frame)
Kristian Høgsberg40979232008-11-25 22:40:39 -0500389{
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500390 wl_list_remove(&client->link);
391 wl_list_insert(client->display->pending_frame_list.prev,
392 &client->link);
393 wl_client_marshal(client, &compositor->base,
394 WL_COMPOSITOR_ACKNOWLEDGE, key, frame);
395}
Kristian Høgsbergfbdbbdc2008-11-28 17:06:06 -0500396
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500397WL_EXPORT int
398wl_display_set_compositor(struct wl_display *display,
399 struct wl_compositor *compositor,
400 const struct wl_compositor_interface *implementation)
401{
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500402 compositor->base.interface = &wl_compositor_interface;
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500403 compositor->base.implementation = (void (**)(void)) implementation;
404
405 wl_display_add_object(display, &compositor->base);
406 if (wl_display_add_global(display, &compositor->base))
407 return -1;
Kristian Høgsberg40979232008-11-25 22:40:39 -0500408
409 return 0;
410}
411
Kristian Høgsberg122912c2008-12-05 11:13:50 -0500412WL_EXPORT struct wl_display *
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400413wl_display_create(void)
414{
415 struct wl_display *display;
416
417 display = malloc(sizeof *display);
418 if (display == NULL)
419 return NULL;
420
421 display->loop = wl_event_loop_create();
422 if (display->loop == NULL) {
423 free(display);
424 return NULL;
425 }
426
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500427 display->objects = wl_hash_create();
428 if (display->objects == NULL) {
429 free(display);
430 return NULL;
431 }
432
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500433 wl_list_init(&display->pending_frame_list);
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500434 wl_list_init(&display->global_list);
Kristian Høgsbergf9bc7952008-11-02 10:12:29 -0500435
Kristian Høgsbergf9212892008-10-11 18:40:23 -0400436 display->client_id_range = 256; /* Gah, arbitrary... */
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400437
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500438 display->id = 1;
Kristian Høgsbergfe831a72008-12-21 21:50:23 -0500439 display->base.interface = &wl_display_interface;
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500440 display->base.implementation = NULL;
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500441 wl_display_add_object(display, &display->base);
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500442 if (wl_display_add_global(display, &display->base)) {
443 wl_event_loop_destroy(display->loop);
444 free(display);
445 return NULL;
446 }
447
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400448 return display;
449}
450
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500451WL_EXPORT void
452wl_display_add_object(struct wl_display *display, struct wl_object *object)
453{
454 object->id = display->id++;
Kristian Høgsberg864c4682008-12-12 11:05:17 -0500455 wl_hash_insert(display->objects, object);
Kristian Høgsberg1e4b86a2008-11-23 23:41:08 -0500456}
457
Kristian Høgsberg14fcff72008-11-23 19:10:23 -0500458WL_EXPORT int
459wl_display_add_global(struct wl_display *display, struct wl_object *object)
460{
461 struct wl_object_ref *ref;
462
463 ref = malloc(sizeof *ref);
464 if (ref == NULL)
465 return -1;
466
467 ref->object = object;
468 wl_list_insert(display->global_list.prev, &ref->link);
469
470 return 0;
471}
472
Kristian Høgsbergcddc0ad2008-11-24 00:31:49 -0500473WL_EXPORT void
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500474wl_surface_post_event(struct wl_surface *surface,
475 struct wl_object *sender,
476 uint32_t event, ...)
Kristian Høgsberg715a0812008-12-10 10:42:04 -0500477{
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500478 va_list ap;
Kristian Høgsberg715a0812008-12-10 10:42:04 -0500479
Kristian Høgsberg5a75c902008-12-10 13:16:50 -0500480 va_start(ap, event);
481 wl_client_vmarshal(surface->client, sender, event, ap);
482 va_end(ap);
Kristian Høgsberg715a0812008-12-10 10:42:04 -0500483}
484
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500485WL_EXPORT void
Kristian Høgsbergfbdbbdc2008-11-28 17:06:06 -0500486wl_display_post_frame(struct wl_display *display,
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500487 struct wl_compositor *compositor,
Kristian Høgsbergfbdbbdc2008-11-28 17:06:06 -0500488 uint32_t frame, uint32_t msecs)
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500489{
490 struct wl_client *client;
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500491
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500492 client = container_of(display->pending_frame_list.next,
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500493 struct wl_client, link);
494
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500495 while (&client->link != &display->pending_frame_list) {
496 wl_client_marshal(client, &compositor->base,
497 WL_COMPOSITOR_FRAME, frame, msecs);
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500498 client = container_of(client->link.next,
499 struct wl_client, link);
500 }
Kristian Høgsberg44f36e32008-11-26 12:57:31 -0500501
Kristian Høgsbergd2412e22008-12-15 20:35:24 -0500502 wl_list_remove(&display->pending_frame_list);
503 wl_list_init(&display->pending_frame_list);
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400504}
505
Kristian Høgsbergb7a01922008-11-08 15:39:41 -0500506WL_EXPORT struct wl_event_loop *
Kristian Høgsberg5ebb3172008-10-11 19:21:35 -0400507wl_display_get_event_loop(struct wl_display *display)
508{
509 return display->loop;
510}
511
Kristian Høgsberg122912c2008-12-05 11:13:50 -0500512WL_EXPORT void
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400513wl_display_run(struct wl_display *display)
514{
515 while (1)
516 wl_event_loop_wait(display->loop);
517}
518
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400519static void
520socket_data(int fd, uint32_t mask, void *data)
521{
522 struct wl_display *display = data;
523 struct sockaddr_un name;
524 socklen_t length;
525 int client_fd;
526
527 length = sizeof name;
528 client_fd = accept (fd, (struct sockaddr *) &name, &length);
529 if (client_fd < 0)
530 fprintf(stderr, "failed to accept\n");
531
532 wl_client_create(display, client_fd);
533}
534
Kristian Høgsberg122912c2008-12-05 11:13:50 -0500535WL_EXPORT int
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500536wl_display_add_socket(struct wl_display *display,
537 const char *name, size_t name_size)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400538{
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500539 struct sockaddr_un addr;
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400540 int sock;
541 socklen_t size;
542
543 sock = socket(PF_LOCAL, SOCK_STREAM, 0);
544 if (sock < 0)
545 return -1;
546
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500547 addr.sun_family = AF_LOCAL;
548 memcpy(addr.sun_path, name, name_size);
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400549
Kristian Høgsbergdc0f3552008-12-07 15:22:22 -0500550 size = offsetof (struct sockaddr_un, sun_path) + name_size;
551 if (bind(sock, (struct sockaddr *) &addr, size) < 0)
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400552 return -1;
553
554 if (listen(sock, 1) < 0)
555 return -1;
556
Kristian Høgsberga67a71a2008-10-07 10:10:36 -0400557 wl_event_loop_add_fd(display->loop, sock,
558 WL_EVENT_READABLE,
Kristian Høgsberg97f1ebe2008-09-30 09:46:10 -0400559 socket_data, display);
560
561 return 0;
562}