2 * Copyright © 2008 Kristian Høgsberg
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.
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
23 * Kristian Høgsberg <krh@bitplanet.net>
24 * Benjamin Franzke <benjaminfranzke@googlemail.com>
34 #include "wayland-server.h"
37 const struct wl_shm_callbacks *callbacks;
40 struct wl_shm_buffer {
41 struct wl_buffer buffer;
49 destroy_buffer(struct wl_resource *resource)
51 struct wl_shm_buffer *buffer =
52 container_of(resource, struct wl_shm_buffer, buffer.resource);
54 munmap(buffer->data, buffer->stride * buffer->buffer.height);
56 buffer->shm->callbacks->buffer_destroyed(&buffer->buffer);
62 shm_buffer_damage(struct wl_client *client, struct wl_resource *resource,
63 int32_t x, int32_t y, int32_t width, int32_t height)
65 struct wl_shm_buffer *buffer = resource->data;
67 buffer->shm->callbacks->buffer_damaged(&buffer->buffer, x, y,
72 shm_buffer_destroy(struct wl_client *client, struct wl_resource *resource)
74 wl_resource_destroy(resource, 0);
77 const static struct wl_buffer_interface shm_buffer_interface = {
82 static struct wl_shm_buffer *
83 wl_shm_buffer_init(struct wl_shm *shm, struct wl_client *client, uint32_t id,
84 int32_t width, int32_t height,
85 int32_t stride, uint32_t format, void *data)
87 struct wl_shm_buffer *buffer;
89 buffer = calloc(1, sizeof *buffer);
93 buffer->buffer.width = width;
94 buffer->buffer.height = height;
95 buffer->format = format;
96 buffer->stride = stride;
99 buffer->buffer.resource.object.id = id;
100 buffer->buffer.resource.object.interface = &wl_buffer_interface;
101 buffer->buffer.resource.object.implementation = (void (**)(void))
102 &shm_buffer_interface;
104 buffer->buffer.resource.data = buffer;
105 buffer->buffer.resource.client = client;
106 buffer->buffer.resource.destroy = destroy_buffer;
110 buffer->shm->callbacks->buffer_created(&buffer->buffer);
116 shm_create_buffer(struct wl_client *client, struct wl_resource *resource,
117 uint32_t id, int fd, int32_t width, int32_t height,
118 uint32_t stride, uint32_t format)
120 struct wl_shm *shm = resource->data;
121 struct wl_shm_buffer *buffer;
126 case WL_SHM_FORMAT_ARGB8888:
127 case WL_SHM_FORMAT_XRGB8888:
130 wl_resource_post_error(resource,
131 WL_SHM_ERROR_INVALID_FORMAT,
137 if (width < 0 || height < 0 || stride < width) {
138 wl_resource_post_error(resource,
139 WL_SHM_ERROR_INVALID_STRIDE,
140 "invalid width, height or stride (%dx%d, %u)",
141 width, height, stride);
146 data = mmap(NULL, stride * height,
147 PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
150 if (data == MAP_FAILED) {
151 wl_resource_post_error(resource,
152 WL_SHM_ERROR_INVALID_FD,
153 "failed mmap fd %d", fd);
157 buffer = wl_shm_buffer_init(shm, client, id,
158 width, height, stride, format, data);
159 if (buffer == NULL) {
160 munmap(data, stride * height);
161 wl_resource_post_no_memory(resource);
165 wl_client_add_resource(client, &buffer->buffer.resource);
168 const static struct wl_shm_interface shm_interface = {
173 bind_shm(struct wl_client *client,
174 void *data, uint32_t version, uint32_t id)
176 struct wl_resource *resource;
178 resource = wl_client_add_object(client, &wl_shm_interface,
179 &shm_interface, id, data);
181 wl_resource_post_event(resource, WL_SHM_FORMAT,
182 WL_SHM_FORMAT_ARGB8888);
183 wl_resource_post_event(resource, WL_SHM_FORMAT,
184 WL_SHM_FORMAT_XRGB8888);
187 WL_EXPORT struct wl_shm *
188 wl_shm_init(struct wl_display *display,
189 const struct wl_shm_callbacks *callbacks)
193 shm = malloc(sizeof *shm);
197 if (!wl_display_add_global(display,
198 &wl_shm_interface, shm, bind_shm)) {
204 shm->callbacks = callbacks;
210 wl_shm_finish(struct wl_shm *shm)
212 /* FIXME: add wl_display_del_{object,global} */
218 wl_buffer_is_shm(struct wl_buffer *buffer)
220 return buffer->resource.object.implementation ==
221 (void (**)(void)) &shm_buffer_interface;
225 wl_shm_buffer_get_stride(struct wl_buffer *buffer_base)
227 struct wl_shm_buffer *buffer = (struct wl_shm_buffer *) buffer_base;
229 if (!wl_buffer_is_shm(buffer_base))
232 return buffer->stride;
236 wl_shm_buffer_get_data(struct wl_buffer *buffer_base)
238 struct wl_shm_buffer *buffer = (struct wl_shm_buffer *) buffer_base;
240 if (!wl_buffer_is_shm(buffer_base))
247 wl_shm_buffer_get_format(struct wl_buffer *buffer_base)
249 struct wl_shm_buffer *buffer = (struct wl_shm_buffer *) buffer_base;
251 return buffer->format;