root / hw / qxl-render.c @ f8f48b69
History | View | Annotate | Download (8.8 kB)
1 | a19cbfb3 | Gerd Hoffmann | /*
|
---|---|---|---|
2 | a19cbfb3 | Gerd Hoffmann | * qxl local rendering (aka display on sdl/vnc)
|
3 | a19cbfb3 | Gerd Hoffmann | *
|
4 | a19cbfb3 | Gerd Hoffmann | * Copyright (C) 2010 Red Hat, Inc.
|
5 | a19cbfb3 | Gerd Hoffmann | *
|
6 | a19cbfb3 | Gerd Hoffmann | * maintained by Gerd Hoffmann <kraxel@redhat.com>
|
7 | a19cbfb3 | Gerd Hoffmann | *
|
8 | a19cbfb3 | Gerd Hoffmann | * This program is free software; you can redistribute it and/or
|
9 | a19cbfb3 | Gerd Hoffmann | * modify it under the terms of the GNU General Public License as
|
10 | a19cbfb3 | Gerd Hoffmann | * published by the Free Software Foundation; either version 2 or
|
11 | a19cbfb3 | Gerd Hoffmann | * (at your option) version 3 of the License.
|
12 | a19cbfb3 | Gerd Hoffmann | *
|
13 | a19cbfb3 | Gerd Hoffmann | * This program is distributed in the hope that it will be useful,
|
14 | a19cbfb3 | Gerd Hoffmann | * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
15 | a19cbfb3 | Gerd Hoffmann | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
16 | a19cbfb3 | Gerd Hoffmann | * GNU General Public License for more details.
|
17 | a19cbfb3 | Gerd Hoffmann | *
|
18 | a19cbfb3 | Gerd Hoffmann | * You should have received a copy of the GNU General Public License
|
19 | a19cbfb3 | Gerd Hoffmann | * along with this program; if not, see <http://www.gnu.org/licenses/>.
|
20 | a19cbfb3 | Gerd Hoffmann | */
|
21 | a19cbfb3 | Gerd Hoffmann | |
22 | a19cbfb3 | Gerd Hoffmann | #include "qxl.h" |
23 | a19cbfb3 | Gerd Hoffmann | |
24 | e2efc0a3 | Gerd Hoffmann | static void qxl_blit(PCIQXLDevice *qxl, QXLRect *rect) |
25 | a19cbfb3 | Gerd Hoffmann | { |
26 | 4c19ebb5 | Alon Levy | uint8_t *src; |
27 | b6e9f637 | Gerd Hoffmann | uint8_t *dst = ds_get_data(qxl->vga.ds); |
28 | a19cbfb3 | Gerd Hoffmann | int len, i;
|
29 | a19cbfb3 | Gerd Hoffmann | |
30 | e2efc0a3 | Gerd Hoffmann | if (is_buffer_shared(qxl->vga.ds->surface)) {
|
31 | 4c19ebb5 | Alon Levy | return;
|
32 | 4c19ebb5 | Alon Levy | } |
33 | 4c19ebb5 | Alon Levy | if (!qxl->guest_primary.data) {
|
34 | d53291cf | Alon Levy | trace_qxl_render_blit_guest_primary_initialized(); |
35 | 4c19ebb5 | Alon Levy | qxl->guest_primary.data = memory_region_get_ram_ptr(&qxl->vga.vram); |
36 | 4c19ebb5 | Alon Levy | } |
37 | d53291cf | Alon Levy | trace_qxl_render_blit(qxl->guest_primary.qxl_stride, |
38 | 4c19ebb5 | Alon Levy | rect->left, rect->right, rect->top, rect->bottom); |
39 | 4c19ebb5 | Alon Levy | src = qxl->guest_primary.data; |
40 | e2efc0a3 | Gerd Hoffmann | if (qxl->guest_primary.qxl_stride < 0) { |
41 | e2efc0a3 | Gerd Hoffmann | /* qxl surface is upside down, walk src scanlines
|
42 | e2efc0a3 | Gerd Hoffmann | * in reverse order to flip it */
|
43 | e2efc0a3 | Gerd Hoffmann | src += (qxl->guest_primary.surface.height - rect->top - 1) *
|
44 | e2efc0a3 | Gerd Hoffmann | qxl->guest_primary.abs_stride; |
45 | e2efc0a3 | Gerd Hoffmann | } else {
|
46 | e2efc0a3 | Gerd Hoffmann | src += rect->top * qxl->guest_primary.abs_stride; |
47 | e2efc0a3 | Gerd Hoffmann | } |
48 | 0e2487bd | Gerd Hoffmann | dst += rect->top * qxl->guest_primary.abs_stride; |
49 | a19cbfb3 | Gerd Hoffmann | src += rect->left * qxl->guest_primary.bytes_pp; |
50 | a19cbfb3 | Gerd Hoffmann | dst += rect->left * qxl->guest_primary.bytes_pp; |
51 | a19cbfb3 | Gerd Hoffmann | len = (rect->right - rect->left) * qxl->guest_primary.bytes_pp; |
52 | a19cbfb3 | Gerd Hoffmann | |
53 | a19cbfb3 | Gerd Hoffmann | for (i = rect->top; i < rect->bottom; i++) {
|
54 | a19cbfb3 | Gerd Hoffmann | memcpy(dst, src, len); |
55 | 0e2487bd | Gerd Hoffmann | dst += qxl->guest_primary.abs_stride; |
56 | e2efc0a3 | Gerd Hoffmann | src += qxl->guest_primary.qxl_stride; |
57 | a19cbfb3 | Gerd Hoffmann | } |
58 | a19cbfb3 | Gerd Hoffmann | } |
59 | a19cbfb3 | Gerd Hoffmann | |
60 | a19cbfb3 | Gerd Hoffmann | void qxl_render_resize(PCIQXLDevice *qxl)
|
61 | a19cbfb3 | Gerd Hoffmann | { |
62 | a19cbfb3 | Gerd Hoffmann | QXLSurfaceCreate *sc = &qxl->guest_primary.surface; |
63 | a19cbfb3 | Gerd Hoffmann | |
64 | 0e2487bd | Gerd Hoffmann | qxl->guest_primary.qxl_stride = sc->stride; |
65 | 0e2487bd | Gerd Hoffmann | qxl->guest_primary.abs_stride = abs(sc->stride); |
66 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.resized++; |
67 | a19cbfb3 | Gerd Hoffmann | switch (sc->format) {
|
68 | a19cbfb3 | Gerd Hoffmann | case SPICE_SURFACE_FMT_16_555:
|
69 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bytes_pp = 2;
|
70 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bits_pp = 15;
|
71 | a19cbfb3 | Gerd Hoffmann | break;
|
72 | a19cbfb3 | Gerd Hoffmann | case SPICE_SURFACE_FMT_16_565:
|
73 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bytes_pp = 2;
|
74 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bits_pp = 16;
|
75 | a19cbfb3 | Gerd Hoffmann | break;
|
76 | a19cbfb3 | Gerd Hoffmann | case SPICE_SURFACE_FMT_32_xRGB:
|
77 | a19cbfb3 | Gerd Hoffmann | case SPICE_SURFACE_FMT_32_ARGB:
|
78 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bytes_pp = 4;
|
79 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bits_pp = 32;
|
80 | a19cbfb3 | Gerd Hoffmann | break;
|
81 | a19cbfb3 | Gerd Hoffmann | default:
|
82 | a19cbfb3 | Gerd Hoffmann | fprintf(stderr, "%s: unhandled format: %x\n", __FUNCTION__,
|
83 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.surface.format); |
84 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bytes_pp = 4;
|
85 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bits_pp = 32;
|
86 | a19cbfb3 | Gerd Hoffmann | break;
|
87 | a19cbfb3 | Gerd Hoffmann | } |
88 | a19cbfb3 | Gerd Hoffmann | } |
89 | a19cbfb3 | Gerd Hoffmann | |
90 | 81fb6f15 | Alon Levy | static void qxl_set_rect_to_surface(PCIQXLDevice *qxl, QXLRect *area) |
91 | 81fb6f15 | Alon Levy | { |
92 | 81fb6f15 | Alon Levy | area->left = 0;
|
93 | 81fb6f15 | Alon Levy | area->right = qxl->guest_primary.surface.width; |
94 | 81fb6f15 | Alon Levy | area->top = 0;
|
95 | 81fb6f15 | Alon Levy | area->bottom = qxl->guest_primary.surface.height; |
96 | 81fb6f15 | Alon Levy | } |
97 | 81fb6f15 | Alon Levy | |
98 | 81fb6f15 | Alon Levy | static void qxl_render_update_area_unlocked(PCIQXLDevice *qxl) |
99 | a19cbfb3 | Gerd Hoffmann | { |
100 | a19cbfb3 | Gerd Hoffmann | VGACommonState *vga = &qxl->vga; |
101 | 81fb6f15 | Alon Levy | int i;
|
102 | a19cbfb3 | Gerd Hoffmann | |
103 | a19cbfb3 | Gerd Hoffmann | if (qxl->guest_primary.resized) {
|
104 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.resized = 0;
|
105 | b1950430 | Avi Kivity | qxl->guest_primary.data = memory_region_get_ram_ptr(&qxl->vga.vram); |
106 | 81fb6f15 | Alon Levy | qxl_set_rect_to_surface(qxl, &qxl->dirty[0]);
|
107 | 81fb6f15 | Alon Levy | qxl->num_dirty_rects = 1;
|
108 | d53291cf | Alon Levy | trace_qxl_render_guest_primary_resized( |
109 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.surface.width, |
110 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.surface.height, |
111 | 0e2487bd | Gerd Hoffmann | qxl->guest_primary.qxl_stride, |
112 | a19cbfb3 | Gerd Hoffmann | qxl->guest_primary.bytes_pp, |
113 | 4c19ebb5 | Alon Levy | qxl->guest_primary.bits_pp); |
114 | 4c19ebb5 | Alon Levy | if (qxl->guest_primary.qxl_stride > 0) { |
115 | 4c19ebb5 | Alon Levy | qemu_free_displaysurface(vga->ds); |
116 | 2f464b5a | Gerd Hoffmann | vga->ds->surface = qemu_create_displaysurface_from |
117 | 2f464b5a | Gerd Hoffmann | (qxl->guest_primary.surface.width, |
118 | 2f464b5a | Gerd Hoffmann | qxl->guest_primary.surface.height, |
119 | 2f464b5a | Gerd Hoffmann | qxl->guest_primary.bits_pp, |
120 | 2f464b5a | Gerd Hoffmann | qxl->guest_primary.abs_stride, |
121 | 2f464b5a | Gerd Hoffmann | qxl->guest_primary.data); |
122 | 4c19ebb5 | Alon Levy | } else {
|
123 | 4c19ebb5 | Alon Levy | qemu_resize_displaysurface(vga->ds, |
124 | 4c19ebb5 | Alon Levy | qxl->guest_primary.surface.width, |
125 | 4c19ebb5 | Alon Levy | qxl->guest_primary.surface.height); |
126 | 4c19ebb5 | Alon Levy | } |
127 | a93a4a22 | Gerd Hoffmann | dpy_gfx_resize(vga->ds); |
128 | a19cbfb3 | Gerd Hoffmann | } |
129 | 81fb6f15 | Alon Levy | for (i = 0; i < qxl->num_dirty_rects; i++) { |
130 | 81fb6f15 | Alon Levy | if (qemu_spice_rect_is_empty(qxl->dirty+i)) {
|
131 | a19cbfb3 | Gerd Hoffmann | break;
|
132 | a19cbfb3 | Gerd Hoffmann | } |
133 | e2efc0a3 | Gerd Hoffmann | qxl_blit(qxl, qxl->dirty+i); |
134 | a93a4a22 | Gerd Hoffmann | dpy_gfx_update(vga->ds, |
135 | a93a4a22 | Gerd Hoffmann | qxl->dirty[i].left, qxl->dirty[i].top, |
136 | a93a4a22 | Gerd Hoffmann | qxl->dirty[i].right - qxl->dirty[i].left, |
137 | a93a4a22 | Gerd Hoffmann | qxl->dirty[i].bottom - qxl->dirty[i].top); |
138 | a19cbfb3 | Gerd Hoffmann | } |
139 | 81fb6f15 | Alon Levy | qxl->num_dirty_rects = 0;
|
140 | 81fb6f15 | Alon Levy | } |
141 | 81fb6f15 | Alon Levy | |
142 | 81fb6f15 | Alon Levy | /*
|
143 | 81fb6f15 | Alon Levy | * use ssd.lock to protect render_update_cookie_num.
|
144 | 81fb6f15 | Alon Levy | * qxl_render_update is called by io thread or vcpu thread, and the completion
|
145 | 81fb6f15 | Alon Levy | * callbacks are called by spice_server thread, defering to bh called from the
|
146 | 81fb6f15 | Alon Levy | * io thread.
|
147 | 81fb6f15 | Alon Levy | */
|
148 | 81fb6f15 | Alon Levy | void qxl_render_update(PCIQXLDevice *qxl)
|
149 | 81fb6f15 | Alon Levy | { |
150 | 81fb6f15 | Alon Levy | QXLCookie *cookie; |
151 | 81fb6f15 | Alon Levy | |
152 | 81fb6f15 | Alon Levy | qemu_mutex_lock(&qxl->ssd.lock); |
153 | 81fb6f15 | Alon Levy | |
154 | 81fb6f15 | Alon Levy | if (!runstate_is_running() || !qxl->guest_primary.commands) {
|
155 | 81fb6f15 | Alon Levy | qxl_render_update_area_unlocked(qxl); |
156 | 81fb6f15 | Alon Levy | qemu_mutex_unlock(&qxl->ssd.lock); |
157 | 81fb6f15 | Alon Levy | return;
|
158 | 81fb6f15 | Alon Levy | } |
159 | 81fb6f15 | Alon Levy | |
160 | 81fb6f15 | Alon Levy | qxl->guest_primary.commands = 0;
|
161 | 81fb6f15 | Alon Levy | qxl->render_update_cookie_num++; |
162 | 81fb6f15 | Alon Levy | qemu_mutex_unlock(&qxl->ssd.lock); |
163 | 81fb6f15 | Alon Levy | cookie = qxl_cookie_new(QXL_COOKIE_TYPE_RENDER_UPDATE_AREA, |
164 | 81fb6f15 | Alon Levy | 0);
|
165 | 81fb6f15 | Alon Levy | qxl_set_rect_to_surface(qxl, &cookie->u.render.area); |
166 | 81fb6f15 | Alon Levy | qxl_spice_update_area(qxl, 0, &cookie->u.render.area, NULL, |
167 | 81fb6f15 | Alon Levy | 0, 1 /* clear_dirty_region */, QXL_ASYNC, cookie); |
168 | 81fb6f15 | Alon Levy | } |
169 | 81fb6f15 | Alon Levy | |
170 | 81fb6f15 | Alon Levy | void qxl_render_update_area_bh(void *opaque) |
171 | 81fb6f15 | Alon Levy | { |
172 | 81fb6f15 | Alon Levy | PCIQXLDevice *qxl = opaque; |
173 | 81fb6f15 | Alon Levy | |
174 | 81fb6f15 | Alon Levy | qemu_mutex_lock(&qxl->ssd.lock); |
175 | 81fb6f15 | Alon Levy | qxl_render_update_area_unlocked(qxl); |
176 | 81fb6f15 | Alon Levy | qemu_mutex_unlock(&qxl->ssd.lock); |
177 | 81fb6f15 | Alon Levy | } |
178 | 81fb6f15 | Alon Levy | |
179 | 81fb6f15 | Alon Levy | void qxl_render_update_area_done(PCIQXLDevice *qxl, QXLCookie *cookie)
|
180 | 81fb6f15 | Alon Levy | { |
181 | 81fb6f15 | Alon Levy | qemu_mutex_lock(&qxl->ssd.lock); |
182 | d53291cf | Alon Levy | trace_qxl_render_update_area_done(cookie); |
183 | 81fb6f15 | Alon Levy | qemu_bh_schedule(qxl->update_area_bh); |
184 | 81fb6f15 | Alon Levy | qxl->render_update_cookie_num--; |
185 | 81fb6f15 | Alon Levy | qemu_mutex_unlock(&qxl->ssd.lock); |
186 | 81fb6f15 | Alon Levy | g_free(cookie); |
187 | a19cbfb3 | Gerd Hoffmann | } |
188 | a19cbfb3 | Gerd Hoffmann | |
189 | a19cbfb3 | Gerd Hoffmann | static QEMUCursor *qxl_cursor(PCIQXLDevice *qxl, QXLCursor *cursor)
|
190 | a19cbfb3 | Gerd Hoffmann | { |
191 | a19cbfb3 | Gerd Hoffmann | QEMUCursor *c; |
192 | a19cbfb3 | Gerd Hoffmann | uint8_t *image, *mask; |
193 | 66d3f196 | Markus Armbruster | size_t size; |
194 | a19cbfb3 | Gerd Hoffmann | |
195 | a19cbfb3 | Gerd Hoffmann | c = cursor_alloc(cursor->header.width, cursor->header.height); |
196 | a19cbfb3 | Gerd Hoffmann | c->hot_x = cursor->header.hot_spot_x; |
197 | a19cbfb3 | Gerd Hoffmann | c->hot_y = cursor->header.hot_spot_y; |
198 | a19cbfb3 | Gerd Hoffmann | switch (cursor->header.type) {
|
199 | a19cbfb3 | Gerd Hoffmann | case SPICE_CURSOR_TYPE_ALPHA:
|
200 | a19cbfb3 | Gerd Hoffmann | size = cursor->header.width * cursor->header.height * sizeof(uint32_t);
|
201 | a19cbfb3 | Gerd Hoffmann | memcpy(c->data, cursor->chunk.data, size); |
202 | a19cbfb3 | Gerd Hoffmann | if (qxl->debug > 2) { |
203 | a19cbfb3 | Gerd Hoffmann | cursor_print_ascii_art(c, "qxl/alpha");
|
204 | a19cbfb3 | Gerd Hoffmann | } |
205 | a19cbfb3 | Gerd Hoffmann | break;
|
206 | a19cbfb3 | Gerd Hoffmann | case SPICE_CURSOR_TYPE_MONO:
|
207 | a19cbfb3 | Gerd Hoffmann | mask = cursor->chunk.data; |
208 | a19cbfb3 | Gerd Hoffmann | image = mask + cursor_get_mono_bpl(c) * c->width; |
209 | a19cbfb3 | Gerd Hoffmann | cursor_set_mono(c, 0xffffff, 0x000000, image, 1, mask); |
210 | a19cbfb3 | Gerd Hoffmann | if (qxl->debug > 2) { |
211 | a19cbfb3 | Gerd Hoffmann | cursor_print_ascii_art(c, "qxl/mono");
|
212 | a19cbfb3 | Gerd Hoffmann | } |
213 | a19cbfb3 | Gerd Hoffmann | break;
|
214 | a19cbfb3 | Gerd Hoffmann | default:
|
215 | a19cbfb3 | Gerd Hoffmann | fprintf(stderr, "%s: not implemented: type %d\n",
|
216 | a19cbfb3 | Gerd Hoffmann | __FUNCTION__, cursor->header.type); |
217 | a19cbfb3 | Gerd Hoffmann | goto fail;
|
218 | a19cbfb3 | Gerd Hoffmann | } |
219 | a19cbfb3 | Gerd Hoffmann | return c;
|
220 | a19cbfb3 | Gerd Hoffmann | |
221 | a19cbfb3 | Gerd Hoffmann | fail:
|
222 | a19cbfb3 | Gerd Hoffmann | cursor_put(c); |
223 | a19cbfb3 | Gerd Hoffmann | return NULL; |
224 | a19cbfb3 | Gerd Hoffmann | } |
225 | a19cbfb3 | Gerd Hoffmann | |
226 | a19cbfb3 | Gerd Hoffmann | |
227 | a19cbfb3 | Gerd Hoffmann | /* called from spice server thread context only */
|
228 | fae2afb1 | Alon Levy | int qxl_render_cursor(PCIQXLDevice *qxl, QXLCommandExt *ext)
|
229 | a19cbfb3 | Gerd Hoffmann | { |
230 | a19cbfb3 | Gerd Hoffmann | QXLCursorCmd *cmd = qxl_phys2virt(qxl, ext->cmd.data, ext->group_id); |
231 | a19cbfb3 | Gerd Hoffmann | QXLCursor *cursor; |
232 | a19cbfb3 | Gerd Hoffmann | QEMUCursor *c; |
233 | a19cbfb3 | Gerd Hoffmann | |
234 | fae2afb1 | Alon Levy | if (!cmd) {
|
235 | fae2afb1 | Alon Levy | return 1; |
236 | fae2afb1 | Alon Levy | } |
237 | fae2afb1 | Alon Levy | |
238 | bf2fde70 | Gerd Hoffmann | if (!dpy_cursor_define_supported(qxl->ssd.ds)) {
|
239 | fae2afb1 | Alon Levy | return 0; |
240 | a19cbfb3 | Gerd Hoffmann | } |
241 | a19cbfb3 | Gerd Hoffmann | |
242 | a19cbfb3 | Gerd Hoffmann | if (qxl->debug > 1 && cmd->type != QXL_CURSOR_MOVE) { |
243 | a19cbfb3 | Gerd Hoffmann | fprintf(stderr, "%s", __FUNCTION__);
|
244 | a19cbfb3 | Gerd Hoffmann | qxl_log_cmd_cursor(qxl, cmd, ext->group_id); |
245 | a19cbfb3 | Gerd Hoffmann | fprintf(stderr, "\n");
|
246 | a19cbfb3 | Gerd Hoffmann | } |
247 | a19cbfb3 | Gerd Hoffmann | switch (cmd->type) {
|
248 | a19cbfb3 | Gerd Hoffmann | case QXL_CURSOR_SET:
|
249 | a19cbfb3 | Gerd Hoffmann | cursor = qxl_phys2virt(qxl, cmd->u.set.shape, ext->group_id); |
250 | fae2afb1 | Alon Levy | if (!cursor) {
|
251 | fae2afb1 | Alon Levy | return 1; |
252 | fae2afb1 | Alon Levy | } |
253 | a19cbfb3 | Gerd Hoffmann | if (cursor->chunk.data_size != cursor->data_size) {
|
254 | a19cbfb3 | Gerd Hoffmann | fprintf(stderr, "%s: multiple chunks\n", __FUNCTION__);
|
255 | fae2afb1 | Alon Levy | return 1; |
256 | a19cbfb3 | Gerd Hoffmann | } |
257 | a19cbfb3 | Gerd Hoffmann | c = qxl_cursor(qxl, cursor); |
258 | a19cbfb3 | Gerd Hoffmann | if (c == NULL) { |
259 | a19cbfb3 | Gerd Hoffmann | c = cursor_builtin_left_ptr(); |
260 | a19cbfb3 | Gerd Hoffmann | } |
261 | 07536094 | Gerd Hoffmann | qemu_mutex_lock(&qxl->ssd.lock); |
262 | 07536094 | Gerd Hoffmann | if (qxl->ssd.cursor) {
|
263 | 07536094 | Gerd Hoffmann | cursor_put(qxl->ssd.cursor); |
264 | 07536094 | Gerd Hoffmann | } |
265 | 07536094 | Gerd Hoffmann | qxl->ssd.cursor = c; |
266 | 07536094 | Gerd Hoffmann | qxl->ssd.mouse_x = cmd->u.set.position.x; |
267 | 07536094 | Gerd Hoffmann | qxl->ssd.mouse_y = cmd->u.set.position.y; |
268 | 07536094 | Gerd Hoffmann | qemu_mutex_unlock(&qxl->ssd.lock); |
269 | a19cbfb3 | Gerd Hoffmann | break;
|
270 | a19cbfb3 | Gerd Hoffmann | case QXL_CURSOR_MOVE:
|
271 | 07536094 | Gerd Hoffmann | qemu_mutex_lock(&qxl->ssd.lock); |
272 | 07536094 | Gerd Hoffmann | qxl->ssd.mouse_x = cmd->u.position.x; |
273 | 07536094 | Gerd Hoffmann | qxl->ssd.mouse_y = cmd->u.position.y; |
274 | 07536094 | Gerd Hoffmann | qemu_mutex_unlock(&qxl->ssd.lock); |
275 | a19cbfb3 | Gerd Hoffmann | break;
|
276 | a19cbfb3 | Gerd Hoffmann | } |
277 | fae2afb1 | Alon Levy | return 0; |
278 | a19cbfb3 | Gerd Hoffmann | } |