xref: /openbmc/linux/drivers/gpu/drm/qxl/qxl_image.c (revision 8be98d2f2a0a262f8bf8a0bc1fdf522b3c7aab17)
1f64122c1SDave Airlie /*
2f64122c1SDave Airlie  * Copyright 2013 Red Hat Inc.
3f64122c1SDave Airlie  *
4f64122c1SDave Airlie  * Permission is hereby granted, free of charge, to any person obtaining a
5f64122c1SDave Airlie  * copy of this software and associated documentation files (the "Software"),
6f64122c1SDave Airlie  * to deal in the Software without restriction, including without limitation
7f64122c1SDave Airlie  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8f64122c1SDave Airlie  * and/or sell copies of the Software, and to permit persons to whom the
9f64122c1SDave Airlie  * Software is furnished to do so, subject to the following conditions:
10f64122c1SDave Airlie  *
11f64122c1SDave Airlie  * The above copyright notice and this permission notice shall be included in
12f64122c1SDave Airlie  * all copies or substantial portions of the Software.
13f64122c1SDave Airlie  *
14f64122c1SDave Airlie  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15f64122c1SDave Airlie  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16f64122c1SDave Airlie  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17f64122c1SDave Airlie  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18f64122c1SDave Airlie  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19f64122c1SDave Airlie  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20f64122c1SDave Airlie  * OTHER DEALINGS IN THE SOFTWARE.
21f64122c1SDave Airlie  *
22f64122c1SDave Airlie  * Authors: Dave Airlie
23f64122c1SDave Airlie  *          Alon Levy
24f64122c1SDave Airlie  */
25f64122c1SDave Airlie 
26f64122c1SDave Airlie #include <linux/gfp.h>
27f64122c1SDave Airlie #include <linux/slab.h>
28f64122c1SDave Airlie 
29f64122c1SDave Airlie #include "qxl_drv.h"
30f64122c1SDave Airlie #include "qxl_object.h"
31f64122c1SDave Airlie 
32f64122c1SDave Airlie static int
qxl_allocate_chunk(struct qxl_device * qdev,struct qxl_release * release,struct qxl_drm_image * image,unsigned int chunk_size)338002db63SDave Airlie qxl_allocate_chunk(struct qxl_device *qdev,
34f64122c1SDave Airlie 		   struct qxl_release *release,
358002db63SDave Airlie 		   struct qxl_drm_image *image,
368002db63SDave Airlie 		   unsigned int chunk_size)
378002db63SDave Airlie {
388002db63SDave Airlie 	struct qxl_drm_chunk *chunk;
398002db63SDave Airlie 	int ret;
408002db63SDave Airlie 
418002db63SDave Airlie 	chunk = kmalloc(sizeof(struct qxl_drm_chunk), GFP_KERNEL);
428002db63SDave Airlie 	if (!chunk)
438002db63SDave Airlie 		return -ENOMEM;
448002db63SDave Airlie 
458002db63SDave Airlie 	ret = qxl_alloc_bo_reserved(qdev, release, chunk_size, &chunk->bo);
468002db63SDave Airlie 	if (ret) {
478002db63SDave Airlie 		kfree(chunk);
488002db63SDave Airlie 		return ret;
498002db63SDave Airlie 	}
508002db63SDave Airlie 
518002db63SDave Airlie 	list_add_tail(&chunk->head, &image->chunk_list);
528002db63SDave Airlie 	return 0;
538002db63SDave Airlie }
548002db63SDave Airlie 
558002db63SDave Airlie int
qxl_image_alloc_objects(struct qxl_device * qdev,struct qxl_release * release,struct qxl_drm_image ** image_ptr,int height,int stride)568002db63SDave Airlie qxl_image_alloc_objects(struct qxl_device *qdev,
578002db63SDave Airlie 			struct qxl_release *release,
588002db63SDave Airlie 			struct qxl_drm_image **image_ptr,
598002db63SDave Airlie 			int height, int stride)
608002db63SDave Airlie {
618002db63SDave Airlie 	struct qxl_drm_image *image;
628002db63SDave Airlie 	int ret;
638002db63SDave Airlie 
648002db63SDave Airlie 	image = kmalloc(sizeof(struct qxl_drm_image), GFP_KERNEL);
658002db63SDave Airlie 	if (!image)
668002db63SDave Airlie 		return -ENOMEM;
678002db63SDave Airlie 
688002db63SDave Airlie 	INIT_LIST_HEAD(&image->chunk_list);
698002db63SDave Airlie 
708002db63SDave Airlie 	ret = qxl_alloc_bo_reserved(qdev, release, sizeof(struct qxl_image), &image->bo);
718002db63SDave Airlie 	if (ret) {
728002db63SDave Airlie 		kfree(image);
738002db63SDave Airlie 		return ret;
748002db63SDave Airlie 	}
758002db63SDave Airlie 
768002db63SDave Airlie 	ret = qxl_allocate_chunk(qdev, release, image, sizeof(struct qxl_data_chunk) + stride * height);
778002db63SDave Airlie 	if (ret) {
788002db63SDave Airlie 		qxl_bo_unref(&image->bo);
798002db63SDave Airlie 		kfree(image);
808002db63SDave Airlie 		return ret;
818002db63SDave Airlie 	}
828002db63SDave Airlie 	*image_ptr = image;
838002db63SDave Airlie 	return 0;
848002db63SDave Airlie }
858002db63SDave Airlie 
qxl_image_free_objects(struct qxl_device * qdev,struct qxl_drm_image * dimage)868002db63SDave Airlie void qxl_image_free_objects(struct qxl_device *qdev, struct qxl_drm_image *dimage)
878002db63SDave Airlie {
888002db63SDave Airlie 	struct qxl_drm_chunk *chunk, *tmp;
898002db63SDave Airlie 
908002db63SDave Airlie 	list_for_each_entry_safe(chunk, tmp, &dimage->chunk_list, head) {
918002db63SDave Airlie 		qxl_bo_unref(&chunk->bo);
928002db63SDave Airlie 		kfree(chunk);
938002db63SDave Airlie 	}
948002db63SDave Airlie 
958002db63SDave Airlie 	qxl_bo_unref(&dimage->bo);
968002db63SDave Airlie 	kfree(dimage);
978002db63SDave Airlie }
988002db63SDave Airlie 
998002db63SDave Airlie static int
qxl_image_init_helper(struct qxl_device * qdev,struct qxl_release * release,struct qxl_drm_image * dimage,const uint8_t * data,int width,int height,int depth,unsigned int hash,int stride)1008002db63SDave Airlie qxl_image_init_helper(struct qxl_device *qdev,
1018002db63SDave Airlie 		      struct qxl_release *release,
1028002db63SDave Airlie 		      struct qxl_drm_image *dimage,
103f64122c1SDave Airlie 		      const uint8_t *data,
104f64122c1SDave Airlie 		      int width, int height,
105f64122c1SDave Airlie 		      int depth, unsigned int hash,
106f64122c1SDave Airlie 		      int stride)
107f64122c1SDave Airlie {
1088002db63SDave Airlie 	struct qxl_drm_chunk *drv_chunk;
109f64122c1SDave Airlie 	struct qxl_image *image;
110f64122c1SDave Airlie 	struct qxl_data_chunk *chunk;
111f64122c1SDave Airlie 	int i;
112f64122c1SDave Airlie 	int chunk_stride;
113f64122c1SDave Airlie 	int linesize = width * depth / 8;
1148002db63SDave Airlie 	struct qxl_bo *chunk_bo, *image_bo;
115f64122c1SDave Airlie 	void *ptr;
116f64122c1SDave Airlie 	/* Chunk */
117f64122c1SDave Airlie 	/* FIXME: Check integer overflow */
118f64122c1SDave Airlie 	/* TODO: variable number of chunks */
1198002db63SDave Airlie 
1208002db63SDave Airlie 	drv_chunk = list_first_entry(&dimage->chunk_list, struct qxl_drm_chunk, head);
1218002db63SDave Airlie 
1228002db63SDave Airlie 	chunk_bo = drv_chunk->bo;
123f64122c1SDave Airlie 	chunk_stride = stride; /* TODO: should use linesize, but it renders
124f64122c1SDave Airlie 				  wrong (check the bitmaps are sent correctly
125f64122c1SDave Airlie 				  first) */
126f64122c1SDave Airlie 
127f64122c1SDave Airlie 	ptr = qxl_bo_kmap_atomic_page(qdev, chunk_bo, 0);
128f64122c1SDave Airlie 	chunk = ptr;
129f64122c1SDave Airlie 	chunk->data_size = height * chunk_stride;
130f64122c1SDave Airlie 	chunk->prev_chunk = 0;
131f64122c1SDave Airlie 	chunk->next_chunk = 0;
132f64122c1SDave Airlie 	qxl_bo_kunmap_atomic_page(qdev, chunk_bo, ptr);
133f64122c1SDave Airlie 
134f64122c1SDave Airlie 	{
135f64122c1SDave Airlie 		void *k_data, *i_data;
136f64122c1SDave Airlie 		int remain;
137f64122c1SDave Airlie 		int page;
138f64122c1SDave Airlie 		int size;
139408799ebSShayenne da Luz Moura 
140f64122c1SDave Airlie 		if (stride == linesize && chunk_stride == stride) {
141f64122c1SDave Airlie 			remain = linesize * height;
142f64122c1SDave Airlie 			page = 0;
143f64122c1SDave Airlie 			i_data = (void *)data;
144f64122c1SDave Airlie 
145f64122c1SDave Airlie 			while (remain > 0) {
146f64122c1SDave Airlie 				ptr = qxl_bo_kmap_atomic_page(qdev, chunk_bo, page << PAGE_SHIFT);
147f64122c1SDave Airlie 
148f64122c1SDave Airlie 				if (page == 0) {
149f64122c1SDave Airlie 					chunk = ptr;
150f64122c1SDave Airlie 					k_data = chunk->data;
151f64122c1SDave Airlie 					size = PAGE_SIZE - offsetof(struct qxl_data_chunk, data);
152f64122c1SDave Airlie 				} else {
153f64122c1SDave Airlie 					k_data = ptr;
154f64122c1SDave Airlie 					size = PAGE_SIZE;
155f64122c1SDave Airlie 				}
156f64122c1SDave Airlie 				size = min(size, remain);
157f64122c1SDave Airlie 
158f64122c1SDave Airlie 				memcpy(k_data, i_data, size);
159f64122c1SDave Airlie 
160f64122c1SDave Airlie 				qxl_bo_kunmap_atomic_page(qdev, chunk_bo, ptr);
161f64122c1SDave Airlie 				i_data += size;
162f64122c1SDave Airlie 				remain -= size;
163f64122c1SDave Airlie 				page++;
164f64122c1SDave Airlie 			}
165f64122c1SDave Airlie 		} else {
1661b000494SShayenne da Luz Moura 			unsigned int page_base, page_offset, out_offset;
167408799ebSShayenne da Luz Moura 
168f64122c1SDave Airlie 			for (i = 0 ; i < height ; ++i) {
169f64122c1SDave Airlie 				i_data = (void *)data + i * stride;
170f64122c1SDave Airlie 				remain = linesize;
171f64122c1SDave Airlie 				out_offset = offsetof(struct qxl_data_chunk, data) + i * chunk_stride;
172f64122c1SDave Airlie 
173f64122c1SDave Airlie 				while (remain > 0) {
174f64122c1SDave Airlie 					page_base = out_offset & PAGE_MASK;
175f64122c1SDave Airlie 					page_offset = offset_in_page(out_offset);
176f64122c1SDave Airlie 					size = min((int)(PAGE_SIZE - page_offset), remain);
177f64122c1SDave Airlie 
178f64122c1SDave Airlie 					ptr = qxl_bo_kmap_atomic_page(qdev, chunk_bo, page_base);
179f64122c1SDave Airlie 					k_data = ptr + page_offset;
180f64122c1SDave Airlie 					memcpy(k_data, i_data, size);
181f64122c1SDave Airlie 					qxl_bo_kunmap_atomic_page(qdev, chunk_bo, ptr);
182f64122c1SDave Airlie 					remain -= size;
183f64122c1SDave Airlie 					i_data += size;
184f64122c1SDave Airlie 					out_offset += size;
185f64122c1SDave Airlie 				}
186f64122c1SDave Airlie 			}
187f64122c1SDave Airlie 		}
188f64122c1SDave Airlie 	}
189*f7ed28e1SGerd Hoffmann 	qxl_bo_vunmap_locked(chunk_bo);
190f64122c1SDave Airlie 
1918002db63SDave Airlie 	image_bo = dimage->bo;
1928002db63SDave Airlie 	ptr = qxl_bo_kmap_atomic_page(qdev, image_bo, 0);
193f64122c1SDave Airlie 	image = ptr;
194f64122c1SDave Airlie 
195f64122c1SDave Airlie 	image->descriptor.id = 0;
196f64122c1SDave Airlie 	image->descriptor.type = SPICE_IMAGE_TYPE_BITMAP;
197f64122c1SDave Airlie 
198f64122c1SDave Airlie 	image->descriptor.flags = 0;
199f64122c1SDave Airlie 	image->descriptor.width = width;
200f64122c1SDave Airlie 	image->descriptor.height = height;
201f64122c1SDave Airlie 
202f64122c1SDave Airlie 	switch (depth) {
203f64122c1SDave Airlie 	case 1:
204f64122c1SDave Airlie 		/* TODO: BE? check by arch? */
205f64122c1SDave Airlie 		image->u.bitmap.format = SPICE_BITMAP_FMT_1BIT_BE;
206f64122c1SDave Airlie 		break;
207f64122c1SDave Airlie 	case 24:
208f64122c1SDave Airlie 		image->u.bitmap.format = SPICE_BITMAP_FMT_24BIT;
209f64122c1SDave Airlie 		break;
210f64122c1SDave Airlie 	case 32:
211f64122c1SDave Airlie 		image->u.bitmap.format = SPICE_BITMAP_FMT_32BIT;
212f64122c1SDave Airlie 		break;
213f64122c1SDave Airlie 	default:
214f64122c1SDave Airlie 		DRM_ERROR("unsupported image bit depth\n");
2155b5703dbSVasily Averin 		qxl_bo_kunmap_atomic_page(qdev, image_bo, ptr);
2165b5703dbSVasily Averin 		return -EINVAL;
217f64122c1SDave Airlie 	}
218f64122c1SDave Airlie 	image->u.bitmap.flags = QXL_BITMAP_TOP_DOWN;
219f64122c1SDave Airlie 	image->u.bitmap.x = width;
220f64122c1SDave Airlie 	image->u.bitmap.y = height;
221f64122c1SDave Airlie 	image->u.bitmap.stride = chunk_stride;
222f64122c1SDave Airlie 	image->u.bitmap.palette = 0;
223f64122c1SDave Airlie 	image->u.bitmap.data = qxl_bo_physical_address(qdev, chunk_bo, 0);
224f64122c1SDave Airlie 
2258002db63SDave Airlie 	qxl_bo_kunmap_atomic_page(qdev, image_bo, ptr);
226f64122c1SDave Airlie 
227f64122c1SDave Airlie 	return 0;
228f64122c1SDave Airlie }
229f64122c1SDave Airlie 
qxl_image_init(struct qxl_device * qdev,struct qxl_release * release,struct qxl_drm_image * dimage,const uint8_t * data,int x,int y,int width,int height,int depth,int stride)2308002db63SDave Airlie int qxl_image_init(struct qxl_device *qdev,
231f64122c1SDave Airlie 		     struct qxl_release *release,
2328002db63SDave Airlie 		     struct qxl_drm_image *dimage,
233f64122c1SDave Airlie 		     const uint8_t *data,
234f64122c1SDave Airlie 		     int x, int y, int width, int height,
235f64122c1SDave Airlie 		     int depth, int stride)
236f64122c1SDave Airlie {
237f64122c1SDave Airlie 	data += y * stride + x * (depth / 8);
2388002db63SDave Airlie 	return qxl_image_init_helper(qdev, release, dimage, data,
239f64122c1SDave Airlie 				       width, height, depth, 0, stride);
240f64122c1SDave Airlie }
241