[PATCH wayland 1/2] shm: Add request for resizing pools

Ander Conselvan de Oliveira ander.conselvan.de.oliveira at intel.com
Tue May 22 05:39:40 PDT 2012


---
 protocol/wayland.xml |   10 ++++++++++
 src/wayland-shm.c    |   33 +++++++++++++++++++++++++++++----
 2 files changed, 39 insertions(+), 4 deletions(-)

diff --git a/protocol/wayland.xml b/protocol/wayland.xml
index d70ced7..2f5110d 100644
--- a/protocol/wayland.xml
+++ b/protocol/wayland.xml
@@ -168,6 +168,16 @@
       <arg name="format" type="uint"/>
     </request>
 
+    <request name="resize">
+      <description summary="change the size of the pool mapping">
+	This request will cause the server to remap the backing memory
+	for the pool from the fd passed when the pool was creating but
+	using the new size.
+      </description>
+
+      <arg name="size" type="int"/>
+    </request>
+
     <request name="destroy" type="destructor">
       <description summary="destroy the pool">
 	Destroy the pool.
diff --git a/src/wayland-shm.c b/src/wayland-shm.c
index 7900ba1..2cc4c73 100644
--- a/src/wayland-shm.c
+++ b/src/wayland-shm.c
@@ -38,13 +38,14 @@ struct wl_shm_pool {
 	int refcount;
 	char *data;
 	int size;
+	int fd;
 };
 
 struct wl_shm_buffer {
 	struct wl_buffer buffer;
 	int32_t stride;
 	uint32_t format;
-	void *data;
+	int offset;
 	struct wl_shm_pool *pool;
 };
 
@@ -56,6 +57,7 @@ shm_pool_unref(struct wl_shm_pool *pool)
 		return;
 
 	munmap(pool->data, pool->size);
+	close(pool->fd);
 	free(pool);
 }
 
@@ -120,7 +122,7 @@ shm_pool_create_buffer(struct wl_client *client, struct wl_resource *resource,
 	buffer->buffer.height = height;
 	buffer->format = format;
 	buffer->stride = stride;
-	buffer->data = pool->data + offset;
+	buffer->offset = offset;
 	buffer->pool = pool;
 	pool->refcount++;
 
@@ -137,6 +139,28 @@ shm_pool_create_buffer(struct wl_client *client, struct wl_resource *resource,
 }
 
 static void
+shm_pool_resize(struct wl_client *client, struct wl_resource *resource,
+		int32_t size)
+{
+	struct wl_shm_pool *pool = resource->data;
+	void *data;
+
+	data = mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED,
+		    pool->fd, 0);
+
+	if (data == MAP_FAILED) {
+		wl_resource_post_error(resource,
+				       WL_SHM_ERROR_INVALID_FD,
+				       "failed mmap fd %d", pool->fd);
+		return;
+	}
+
+	munmap(pool->data, pool->size);
+	pool->data = data;
+	pool->size = size;
+}
+
+static void
 destroy_pool(struct wl_resource *resource)
 {
 	struct wl_shm_pool *pool = resource->data;
@@ -152,6 +176,7 @@ shm_pool_destroy(struct wl_client *client, struct wl_resource *resource)
 
 struct wl_shm_pool_interface shm_pool_interface = {
 	shm_pool_create_buffer,
+	shm_pool_resize,
 	shm_pool_destroy
 };
 
@@ -177,10 +202,10 @@ shm_create_pool(struct wl_client *client, struct wl_resource *resource,
 	}
 
 	pool->refcount = 1;
+	pool->fd = fd;
 	pool->size = size;
 	pool->data = mmap(NULL, size,
 			  PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
-	close(fd);
 	if (pool->data == MAP_FAILED) {
 		wl_resource_post_error(resource,
 				       WL_SHM_ERROR_INVALID_FD,
@@ -252,7 +277,7 @@ wl_shm_buffer_get_data(struct wl_buffer *buffer_base)
 	if (!wl_buffer_is_shm(buffer_base))
 		return NULL;
 
-	return buffer->data;
+	return buffer->pool->data + buffer->offset;
 }
 
 WL_EXPORT uint32_t
-- 
1.7.4.1



More information about the wayland-devel mailing list