xref: /OK3568_Linux_fs/kernel/drivers/gpu/drm/qxl/qxl_image.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun /*
2*4882a593Smuzhiyun  * Copyright 2013 Red Hat Inc.
3*4882a593Smuzhiyun  *
4*4882a593Smuzhiyun  * Permission is hereby granted, free of charge, to any person obtaining a
5*4882a593Smuzhiyun  * copy of this software and associated documentation files (the "Software"),
6*4882a593Smuzhiyun  * to deal in the Software without restriction, including without limitation
7*4882a593Smuzhiyun  * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*4882a593Smuzhiyun  * and/or sell copies of the Software, and to permit persons to whom the
9*4882a593Smuzhiyun  * Software is furnished to do so, subject to the following conditions:
10*4882a593Smuzhiyun  *
11*4882a593Smuzhiyun  * The above copyright notice and this permission notice shall be included in
12*4882a593Smuzhiyun  * all copies or substantial portions of the Software.
13*4882a593Smuzhiyun  *
14*4882a593Smuzhiyun  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
15*4882a593Smuzhiyun  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
16*4882a593Smuzhiyun  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL
17*4882a593Smuzhiyun  * THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
18*4882a593Smuzhiyun  * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
19*4882a593Smuzhiyun  * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
20*4882a593Smuzhiyun  * OTHER DEALINGS IN THE SOFTWARE.
21*4882a593Smuzhiyun  *
22*4882a593Smuzhiyun  * Authors: Dave Airlie
23*4882a593Smuzhiyun  *          Alon Levy
24*4882a593Smuzhiyun  */
25*4882a593Smuzhiyun 
26*4882a593Smuzhiyun #include <linux/gfp.h>
27*4882a593Smuzhiyun #include <linux/slab.h>
28*4882a593Smuzhiyun 
29*4882a593Smuzhiyun #include "qxl_drv.h"
30*4882a593Smuzhiyun #include "qxl_object.h"
31*4882a593Smuzhiyun 
32*4882a593Smuzhiyun static int
qxl_allocate_chunk(struct qxl_device * qdev,struct qxl_release * release,struct qxl_drm_image * image,unsigned int chunk_size)33*4882a593Smuzhiyun qxl_allocate_chunk(struct qxl_device *qdev,
34*4882a593Smuzhiyun 		   struct qxl_release *release,
35*4882a593Smuzhiyun 		   struct qxl_drm_image *image,
36*4882a593Smuzhiyun 		   unsigned int chunk_size)
37*4882a593Smuzhiyun {
38*4882a593Smuzhiyun 	struct qxl_drm_chunk *chunk;
39*4882a593Smuzhiyun 	int ret;
40*4882a593Smuzhiyun 
41*4882a593Smuzhiyun 	chunk = kmalloc(sizeof(struct qxl_drm_chunk), GFP_KERNEL);
42*4882a593Smuzhiyun 	if (!chunk)
43*4882a593Smuzhiyun 		return -ENOMEM;
44*4882a593Smuzhiyun 
45*4882a593Smuzhiyun 	ret = qxl_alloc_bo_reserved(qdev, release, chunk_size, &chunk->bo);
46*4882a593Smuzhiyun 	if (ret) {
47*4882a593Smuzhiyun 		kfree(chunk);
48*4882a593Smuzhiyun 		return ret;
49*4882a593Smuzhiyun 	}
50*4882a593Smuzhiyun 
51*4882a593Smuzhiyun 	list_add_tail(&chunk->head, &image->chunk_list);
52*4882a593Smuzhiyun 	return 0;
53*4882a593Smuzhiyun }
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun int
qxl_image_alloc_objects(struct qxl_device * qdev,struct qxl_release * release,struct qxl_drm_image ** image_ptr,int height,int stride)56*4882a593Smuzhiyun qxl_image_alloc_objects(struct qxl_device *qdev,
57*4882a593Smuzhiyun 			struct qxl_release *release,
58*4882a593Smuzhiyun 			struct qxl_drm_image **image_ptr,
59*4882a593Smuzhiyun 			int height, int stride)
60*4882a593Smuzhiyun {
61*4882a593Smuzhiyun 	struct qxl_drm_image *image;
62*4882a593Smuzhiyun 	int ret;
63*4882a593Smuzhiyun 
64*4882a593Smuzhiyun 	image = kmalloc(sizeof(struct qxl_drm_image), GFP_KERNEL);
65*4882a593Smuzhiyun 	if (!image)
66*4882a593Smuzhiyun 		return -ENOMEM;
67*4882a593Smuzhiyun 
68*4882a593Smuzhiyun 	INIT_LIST_HEAD(&image->chunk_list);
69*4882a593Smuzhiyun 
70*4882a593Smuzhiyun 	ret = qxl_alloc_bo_reserved(qdev, release, sizeof(struct qxl_image), &image->bo);
71*4882a593Smuzhiyun 	if (ret) {
72*4882a593Smuzhiyun 		kfree(image);
73*4882a593Smuzhiyun 		return ret;
74*4882a593Smuzhiyun 	}
75*4882a593Smuzhiyun 
76*4882a593Smuzhiyun 	ret = qxl_allocate_chunk(qdev, release, image, sizeof(struct qxl_data_chunk) + stride * height);
77*4882a593Smuzhiyun 	if (ret) {
78*4882a593Smuzhiyun 		qxl_bo_unref(&image->bo);
79*4882a593Smuzhiyun 		kfree(image);
80*4882a593Smuzhiyun 		return ret;
81*4882a593Smuzhiyun 	}
82*4882a593Smuzhiyun 	*image_ptr = image;
83*4882a593Smuzhiyun 	return 0;
84*4882a593Smuzhiyun }
85*4882a593Smuzhiyun 
qxl_image_free_objects(struct qxl_device * qdev,struct qxl_drm_image * dimage)86*4882a593Smuzhiyun void qxl_image_free_objects(struct qxl_device *qdev, struct qxl_drm_image *dimage)
87*4882a593Smuzhiyun {
88*4882a593Smuzhiyun 	struct qxl_drm_chunk *chunk, *tmp;
89*4882a593Smuzhiyun 
90*4882a593Smuzhiyun 	list_for_each_entry_safe(chunk, tmp, &dimage->chunk_list, head) {
91*4882a593Smuzhiyun 		qxl_bo_unref(&chunk->bo);
92*4882a593Smuzhiyun 		kfree(chunk);
93*4882a593Smuzhiyun 	}
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	qxl_bo_unref(&dimage->bo);
96*4882a593Smuzhiyun 	kfree(dimage);
97*4882a593Smuzhiyun }
98*4882a593Smuzhiyun 
99*4882a593Smuzhiyun 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)100*4882a593Smuzhiyun qxl_image_init_helper(struct qxl_device *qdev,
101*4882a593Smuzhiyun 		      struct qxl_release *release,
102*4882a593Smuzhiyun 		      struct qxl_drm_image *dimage,
103*4882a593Smuzhiyun 		      const uint8_t *data,
104*4882a593Smuzhiyun 		      int width, int height,
105*4882a593Smuzhiyun 		      int depth, unsigned int hash,
106*4882a593Smuzhiyun 		      int stride)
107*4882a593Smuzhiyun {
108*4882a593Smuzhiyun 	struct qxl_drm_chunk *drv_chunk;
109*4882a593Smuzhiyun 	struct qxl_image *image;
110*4882a593Smuzhiyun 	struct qxl_data_chunk *chunk;
111*4882a593Smuzhiyun 	int i;
112*4882a593Smuzhiyun 	int chunk_stride;
113*4882a593Smuzhiyun 	int linesize = width * depth / 8;
114*4882a593Smuzhiyun 	struct qxl_bo *chunk_bo, *image_bo;
115*4882a593Smuzhiyun 	void *ptr;
116*4882a593Smuzhiyun 	/* Chunk */
117*4882a593Smuzhiyun 	/* FIXME: Check integer overflow */
118*4882a593Smuzhiyun 	/* TODO: variable number of chunks */
119*4882a593Smuzhiyun 
120*4882a593Smuzhiyun 	drv_chunk = list_first_entry(&dimage->chunk_list, struct qxl_drm_chunk, head);
121*4882a593Smuzhiyun 
122*4882a593Smuzhiyun 	chunk_bo = drv_chunk->bo;
123*4882a593Smuzhiyun 	chunk_stride = stride; /* TODO: should use linesize, but it renders
124*4882a593Smuzhiyun 				  wrong (check the bitmaps are sent correctly
125*4882a593Smuzhiyun 				  first) */
126*4882a593Smuzhiyun 
127*4882a593Smuzhiyun 	ptr = qxl_bo_kmap_atomic_page(qdev, chunk_bo, 0);
128*4882a593Smuzhiyun 	chunk = ptr;
129*4882a593Smuzhiyun 	chunk->data_size = height * chunk_stride;
130*4882a593Smuzhiyun 	chunk->prev_chunk = 0;
131*4882a593Smuzhiyun 	chunk->next_chunk = 0;
132*4882a593Smuzhiyun 	qxl_bo_kunmap_atomic_page(qdev, chunk_bo, ptr);
133*4882a593Smuzhiyun 
134*4882a593Smuzhiyun 	{
135*4882a593Smuzhiyun 		void *k_data, *i_data;
136*4882a593Smuzhiyun 		int remain;
137*4882a593Smuzhiyun 		int page;
138*4882a593Smuzhiyun 		int size;
139*4882a593Smuzhiyun 
140*4882a593Smuzhiyun 		if (stride == linesize && chunk_stride == stride) {
141*4882a593Smuzhiyun 			remain = linesize * height;
142*4882a593Smuzhiyun 			page = 0;
143*4882a593Smuzhiyun 			i_data = (void *)data;
144*4882a593Smuzhiyun 
145*4882a593Smuzhiyun 			while (remain > 0) {
146*4882a593Smuzhiyun 				ptr = qxl_bo_kmap_atomic_page(qdev, chunk_bo, page << PAGE_SHIFT);
147*4882a593Smuzhiyun 
148*4882a593Smuzhiyun 				if (page == 0) {
149*4882a593Smuzhiyun 					chunk = ptr;
150*4882a593Smuzhiyun 					k_data = chunk->data;
151*4882a593Smuzhiyun 					size = PAGE_SIZE - offsetof(struct qxl_data_chunk, data);
152*4882a593Smuzhiyun 				} else {
153*4882a593Smuzhiyun 					k_data = ptr;
154*4882a593Smuzhiyun 					size = PAGE_SIZE;
155*4882a593Smuzhiyun 				}
156*4882a593Smuzhiyun 				size = min(size, remain);
157*4882a593Smuzhiyun 
158*4882a593Smuzhiyun 				memcpy(k_data, i_data, size);
159*4882a593Smuzhiyun 
160*4882a593Smuzhiyun 				qxl_bo_kunmap_atomic_page(qdev, chunk_bo, ptr);
161*4882a593Smuzhiyun 				i_data += size;
162*4882a593Smuzhiyun 				remain -= size;
163*4882a593Smuzhiyun 				page++;
164*4882a593Smuzhiyun 			}
165*4882a593Smuzhiyun 		} else {
166*4882a593Smuzhiyun 			unsigned int page_base, page_offset, out_offset;
167*4882a593Smuzhiyun 
168*4882a593Smuzhiyun 			for (i = 0 ; i < height ; ++i) {
169*4882a593Smuzhiyun 				i_data = (void *)data + i * stride;
170*4882a593Smuzhiyun 				remain = linesize;
171*4882a593Smuzhiyun 				out_offset = offsetof(struct qxl_data_chunk, data) + i * chunk_stride;
172*4882a593Smuzhiyun 
173*4882a593Smuzhiyun 				while (remain > 0) {
174*4882a593Smuzhiyun 					page_base = out_offset & PAGE_MASK;
175*4882a593Smuzhiyun 					page_offset = offset_in_page(out_offset);
176*4882a593Smuzhiyun 					size = min((int)(PAGE_SIZE - page_offset), remain);
177*4882a593Smuzhiyun 
178*4882a593Smuzhiyun 					ptr = qxl_bo_kmap_atomic_page(qdev, chunk_bo, page_base);
179*4882a593Smuzhiyun 					k_data = ptr + page_offset;
180*4882a593Smuzhiyun 					memcpy(k_data, i_data, size);
181*4882a593Smuzhiyun 					qxl_bo_kunmap_atomic_page(qdev, chunk_bo, ptr);
182*4882a593Smuzhiyun 					remain -= size;
183*4882a593Smuzhiyun 					i_data += size;
184*4882a593Smuzhiyun 					out_offset += size;
185*4882a593Smuzhiyun 				}
186*4882a593Smuzhiyun 			}
187*4882a593Smuzhiyun 		}
188*4882a593Smuzhiyun 	}
189*4882a593Smuzhiyun 	qxl_bo_kunmap(chunk_bo);
190*4882a593Smuzhiyun 
191*4882a593Smuzhiyun 	image_bo = dimage->bo;
192*4882a593Smuzhiyun 	ptr = qxl_bo_kmap_atomic_page(qdev, image_bo, 0);
193*4882a593Smuzhiyun 	image = ptr;
194*4882a593Smuzhiyun 
195*4882a593Smuzhiyun 	image->descriptor.id = 0;
196*4882a593Smuzhiyun 	image->descriptor.type = SPICE_IMAGE_TYPE_BITMAP;
197*4882a593Smuzhiyun 
198*4882a593Smuzhiyun 	image->descriptor.flags = 0;
199*4882a593Smuzhiyun 	image->descriptor.width = width;
200*4882a593Smuzhiyun 	image->descriptor.height = height;
201*4882a593Smuzhiyun 
202*4882a593Smuzhiyun 	switch (depth) {
203*4882a593Smuzhiyun 	case 1:
204*4882a593Smuzhiyun 		/* TODO: BE? check by arch? */
205*4882a593Smuzhiyun 		image->u.bitmap.format = SPICE_BITMAP_FMT_1BIT_BE;
206*4882a593Smuzhiyun 		break;
207*4882a593Smuzhiyun 	case 24:
208*4882a593Smuzhiyun 		image->u.bitmap.format = SPICE_BITMAP_FMT_24BIT;
209*4882a593Smuzhiyun 		break;
210*4882a593Smuzhiyun 	case 32:
211*4882a593Smuzhiyun 		image->u.bitmap.format = SPICE_BITMAP_FMT_32BIT;
212*4882a593Smuzhiyun 		break;
213*4882a593Smuzhiyun 	default:
214*4882a593Smuzhiyun 		DRM_ERROR("unsupported image bit depth\n");
215*4882a593Smuzhiyun 		qxl_bo_kunmap_atomic_page(qdev, image_bo, ptr);
216*4882a593Smuzhiyun 		return -EINVAL;
217*4882a593Smuzhiyun 	}
218*4882a593Smuzhiyun 	image->u.bitmap.flags = QXL_BITMAP_TOP_DOWN;
219*4882a593Smuzhiyun 	image->u.bitmap.x = width;
220*4882a593Smuzhiyun 	image->u.bitmap.y = height;
221*4882a593Smuzhiyun 	image->u.bitmap.stride = chunk_stride;
222*4882a593Smuzhiyun 	image->u.bitmap.palette = 0;
223*4882a593Smuzhiyun 	image->u.bitmap.data = qxl_bo_physical_address(qdev, chunk_bo, 0);
224*4882a593Smuzhiyun 
225*4882a593Smuzhiyun 	qxl_bo_kunmap_atomic_page(qdev, image_bo, ptr);
226*4882a593Smuzhiyun 
227*4882a593Smuzhiyun 	return 0;
228*4882a593Smuzhiyun }
229*4882a593Smuzhiyun 
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)230*4882a593Smuzhiyun int qxl_image_init(struct qxl_device *qdev,
231*4882a593Smuzhiyun 		     struct qxl_release *release,
232*4882a593Smuzhiyun 		     struct qxl_drm_image *dimage,
233*4882a593Smuzhiyun 		     const uint8_t *data,
234*4882a593Smuzhiyun 		     int x, int y, int width, int height,
235*4882a593Smuzhiyun 		     int depth, int stride)
236*4882a593Smuzhiyun {
237*4882a593Smuzhiyun 	data += y * stride + x * (depth / 8);
238*4882a593Smuzhiyun 	return qxl_image_init_helper(qdev, release, dimage, data,
239*4882a593Smuzhiyun 				       width, height, depth, 0, stride);
240*4882a593Smuzhiyun }
241