Statistics
| Branch: | Revision:

root / sdl.c @ 70240ca6

History | View | Annotate | Download (21.7 kB)

1 0f0b7264 bellard
/*
2 0f0b7264 bellard
 * QEMU SDL display driver
3 5fafdf24 ths
 *
4 0f0b7264 bellard
 * Copyright (c) 2003 Fabrice Bellard
5 5fafdf24 ths
 *
6 0f0b7264 bellard
 * Permission is hereby granted, free of charge, to any person obtaining a copy
7 0f0b7264 bellard
 * of this software and associated documentation files (the "Software"), to deal
8 0f0b7264 bellard
 * in the Software without restriction, including without limitation the rights
9 0f0b7264 bellard
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 0f0b7264 bellard
 * copies of the Software, and to permit persons to whom the Software is
11 0f0b7264 bellard
 * furnished to do so, subject to the following conditions:
12 0f0b7264 bellard
 *
13 0f0b7264 bellard
 * The above copyright notice and this permission notice shall be included in
14 0f0b7264 bellard
 * all copies or substantial portions of the Software.
15 0f0b7264 bellard
 *
16 0f0b7264 bellard
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 0f0b7264 bellard
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 0f0b7264 bellard
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19 0f0b7264 bellard
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 0f0b7264 bellard
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 0f0b7264 bellard
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 0f0b7264 bellard
 * THE SOFTWARE.
23 0f0b7264 bellard
 */
24 0f0b7264 bellard
#include <SDL.h>
25 c9985aa8 aliguori
#include <SDL_syswm.h>
26 0f0b7264 bellard
27 67b915a5 bellard
#ifndef _WIN32
28 67b915a5 bellard
#include <signal.h>
29 67b915a5 bellard
#endif
30 0f0b7264 bellard
31 511d2b14 blueswir1
#include "qemu-common.h"
32 511d2b14 blueswir1
#include "console.h"
33 511d2b14 blueswir1
#include "sysemu.h"
34 511d2b14 blueswir1
#include "x_keymap.h"
35 511d2b14 blueswir1
36 7d957bd8 aliguori
static DisplayChangeListener *dcl;
37 7d957bd8 aliguori
static SDL_Surface *real_screen;
38 7d957bd8 aliguori
static SDL_Surface *guest_screen = NULL;
39 0f0b7264 bellard
static int gui_grab; /* if true, all keyboard/mouse events are grabbed */
40 8a7ddc38 bellard
static int last_vm_running;
41 8e9c4afe bellard
static int gui_saved_grab;
42 8e9c4afe bellard
static int gui_fullscreen;
43 43523e93 ths
static int gui_noframe;
44 8e9c4afe bellard
static int gui_key_modifier_pressed;
45 8e9c4afe bellard
static int gui_keysym;
46 d63d307f bellard
static int gui_fullscreen_initial_grab;
47 32ff25bf bellard
static int gui_grab_code = KMOD_LALT | KMOD_LCTRL;
48 32ff25bf bellard
static uint8_t modifiers_state[256];
49 09b26c5e bellard
static int width, height;
50 09b26c5e bellard
static SDL_Cursor *sdl_cursor_normal;
51 09b26c5e bellard
static SDL_Cursor *sdl_cursor_hidden;
52 09b26c5e bellard
static int absolute_enabled = 0;
53 d34cab9f ths
static int guest_cursor = 0;
54 d34cab9f ths
static int guest_x, guest_y;
55 d34cab9f ths
static SDL_Cursor *guest_sprite = 0;
56 0f0b7264 bellard
57 0f0b7264 bellard
static void sdl_update(DisplayState *ds, int x, int y, int w, int h)
58 0f0b7264 bellard
{
59 7d957bd8 aliguori
    SDL_Rect rec;
60 7d957bd8 aliguori
    rec.x = x;
61 7d957bd8 aliguori
    rec.y = y;
62 7d957bd8 aliguori
    rec.w = w;
63 7d957bd8 aliguori
    rec.h = h;
64 898712a8 bellard
    //    printf("updating x=%d y=%d w=%d h=%d\n", x, y, w, h);
65 7d957bd8 aliguori
66 7d957bd8 aliguori
    SDL_BlitSurface(guest_screen, &rec, real_screen, &rec);
67 6e60065f pbrook
    SDL_UpdateRect(real_screen, x, y, w, h);
68 7d957bd8 aliguori
}
69 7d957bd8 aliguori
70 7d957bd8 aliguori
static void sdl_setdata(DisplayState *ds)
71 7d957bd8 aliguori
{
72 7d957bd8 aliguori
    SDL_Rect rec;
73 7d957bd8 aliguori
    rec.x = 0;
74 7d957bd8 aliguori
    rec.y = 0;
75 7d957bd8 aliguori
    rec.w = real_screen->w;
76 7d957bd8 aliguori
    rec.h = real_screen->h;
77 7d957bd8 aliguori
78 7d957bd8 aliguori
    if (guest_screen != NULL) SDL_FreeSurface(guest_screen);
79 7d957bd8 aliguori
80 7d957bd8 aliguori
    guest_screen = SDL_CreateRGBSurfaceFrom(ds_get_data(ds), ds_get_width(ds), ds_get_height(ds),
81 7d957bd8 aliguori
                                            ds_get_bits_per_pixel(ds), ds_get_linesize(ds),
82 7d957bd8 aliguori
                                            ds->surface->pf.rmask, ds->surface->pf.gmask,
83 7d957bd8 aliguori
                                            ds->surface->pf.bmask, ds->surface->pf.amask);
84 0f0b7264 bellard
}
85 0f0b7264 bellard
86 7d957bd8 aliguori
static void sdl_resize(DisplayState *ds)
87 0f0b7264 bellard
{
88 0f0b7264 bellard
    int flags;
89 0f0b7264 bellard
90 0f0b7264 bellard
    //    printf("resizing to %d %d\n", w, h);
91 0f0b7264 bellard
92 0f0b7264 bellard
    flags = SDL_HWSURFACE|SDL_ASYNCBLIT|SDL_HWACCEL;
93 8e9c4afe bellard
    if (gui_fullscreen)
94 8e9c4afe bellard
        flags |= SDL_FULLSCREEN;
95 43523e93 ths
    if (gui_noframe)
96 43523e93 ths
        flags |= SDL_NOFRAME;
97 9903da21 bellard
98 8bf66d40 balrog
    width = ds_get_width(ds);
99 8bf66d40 balrog
    height = ds_get_height(ds);
100 8bf66d40 balrog
    real_screen = SDL_SetVideoMode(width, height, 0, flags);
101 7d957bd8 aliguori
    if (!real_screen) {
102 0f0b7264 bellard
        fprintf(stderr, "Could not open SDL display\n");
103 0f0b7264 bellard
        exit(1);
104 0f0b7264 bellard
    }
105 7d957bd8 aliguori
106 7d957bd8 aliguori
    sdl_setdata(ds);
107 0f0b7264 bellard
}
108 0f0b7264 bellard
109 3d11d0eb bellard
/* generic keyboard conversion */
110 e58d12ed bellard
111 3d11d0eb bellard
#include "sdl_keysym.h"
112 3d11d0eb bellard
113 3d11d0eb bellard
static kbd_layout_t *kbd_layout = NULL;
114 3d11d0eb bellard
115 3d11d0eb bellard
static uint8_t sdl_keyevent_to_keycode_generic(const SDL_KeyboardEvent *ev)
116 e58d12ed bellard
{
117 3d11d0eb bellard
    int keysym;
118 3d11d0eb bellard
    /* workaround for X11+SDL bug with AltGR */
119 3d11d0eb bellard
    keysym = ev->keysym.sym;
120 3d11d0eb bellard
    if (keysym == 0 && ev->keysym.scancode == 113)
121 3d11d0eb bellard
        keysym = SDLK_MODE;
122 60659e3b bellard
    /* For Japanese key '\' and '|' */
123 60659e3b bellard
    if (keysym == 92 && ev->keysym.scancode == 133) {
124 60659e3b bellard
        keysym = 0xa5;
125 60659e3b bellard
    }
126 3d11d0eb bellard
    return keysym2scancode(kbd_layout, keysym);
127 e58d12ed bellard
}
128 e58d12ed bellard
129 3d11d0eb bellard
/* specific keyboard conversions from scan codes */
130 3d11d0eb bellard
131 3d11d0eb bellard
#if defined(_WIN32)
132 e58d12ed bellard
133 e58d12ed bellard
static uint8_t sdl_keyevent_to_keycode(const SDL_KeyboardEvent *ev)
134 e58d12ed bellard
{
135 e58d12ed bellard
    return ev->keysym.scancode;
136 e58d12ed bellard
}
137 e58d12ed bellard
138 e58d12ed bellard
#else
139 e58d12ed bellard
140 5368a422 aliguori
#if defined(SDL_VIDEO_DRIVER_X11)
141 5368a422 aliguori
#include <X11/XKBlib.h>
142 5368a422 aliguori
143 5368a422 aliguori
static int check_for_evdev(void)
144 5368a422 aliguori
{
145 5368a422 aliguori
    SDL_SysWMinfo info;
146 5368a422 aliguori
    XkbDescPtr desc;
147 5368a422 aliguori
    int has_evdev = 0;
148 5368a422 aliguori
    const char *keycodes;
149 5368a422 aliguori
150 5368a422 aliguori
    SDL_VERSION(&info.version);
151 5368a422 aliguori
    if (!SDL_GetWMInfo(&info))
152 5368a422 aliguori
        return 0;
153 5368a422 aliguori
154 5368a422 aliguori
    desc = XkbGetKeyboard(info.info.x11.display,
155 5368a422 aliguori
                          XkbGBN_AllComponentsMask,
156 5368a422 aliguori
                          XkbUseCoreKbd);
157 5368a422 aliguori
    if (desc == NULL || desc->names == NULL)
158 5368a422 aliguori
        return 0;
159 5368a422 aliguori
160 5368a422 aliguori
    keycodes = XGetAtomName(info.info.x11.display, desc->names->keycodes);
161 5368a422 aliguori
    if (keycodes == NULL)
162 5368a422 aliguori
        fprintf(stderr, "could not lookup keycode name\n");
163 5368a422 aliguori
    else if (strstart(keycodes, "evdev_", NULL))
164 5368a422 aliguori
        has_evdev = 1;
165 5368a422 aliguori
    else if (!strstart(keycodes, "xfree86_", NULL))
166 5368a422 aliguori
        fprintf(stderr,
167 5368a422 aliguori
                "unknown keycodes `%s', please report to qemu-devel@nongnu.org\n",
168 5368a422 aliguori
                keycodes);
169 5368a422 aliguori
170 5368a422 aliguori
    XkbFreeClientMap(desc, XkbGBN_AllComponentsMask, True);
171 5368a422 aliguori
172 5368a422 aliguori
    return has_evdev;
173 5368a422 aliguori
}
174 5368a422 aliguori
#else
175 5368a422 aliguori
static int check_for_evdev(void)
176 5368a422 aliguori
{
177 5368a422 aliguori
        return 0;
178 5368a422 aliguori
}
179 5368a422 aliguori
#endif
180 5368a422 aliguori
181 e58d12ed bellard
static uint8_t sdl_keyevent_to_keycode(const SDL_KeyboardEvent *ev)
182 e58d12ed bellard
{
183 e58d12ed bellard
    int keycode;
184 5368a422 aliguori
    static int has_evdev = -1;
185 5368a422 aliguori
186 5368a422 aliguori
    if (has_evdev == -1)
187 5368a422 aliguori
        has_evdev = check_for_evdev();
188 e58d12ed bellard
189 e58d12ed bellard
    keycode = ev->keysym.scancode;
190 e58d12ed bellard
191 e58d12ed bellard
    if (keycode < 9) {
192 e58d12ed bellard
        keycode = 0;
193 e58d12ed bellard
    } else if (keycode < 97) {
194 e58d12ed bellard
        keycode -= 8; /* just an offset */
195 5368a422 aliguori
    } else if (keycode < 158) {
196 e58d12ed bellard
        /* use conversion table */
197 5368a422 aliguori
        if (has_evdev)
198 5368a422 aliguori
            keycode = translate_evdev_keycode(keycode - 97);
199 5368a422 aliguori
        else
200 5368a422 aliguori
            keycode = translate_xfree86_keycode(keycode - 97);
201 5368a422 aliguori
    } else if (keycode == 208) { /* Hiragana_Katakana */
202 5368a422 aliguori
        keycode = 0x70;
203 5368a422 aliguori
    } else if (keycode == 211) { /* backslash */
204 5368a422 aliguori
        keycode = 0x73;
205 e58d12ed bellard
    } else {
206 e58d12ed bellard
        keycode = 0;
207 e58d12ed bellard
    }
208 e58d12ed bellard
    return keycode;
209 e58d12ed bellard
}
210 e58d12ed bellard
211 e58d12ed bellard
#endif
212 e58d12ed bellard
213 32ff25bf bellard
static void reset_keys(void)
214 32ff25bf bellard
{
215 32ff25bf bellard
    int i;
216 32ff25bf bellard
    for(i = 0; i < 256; i++) {
217 32ff25bf bellard
        if (modifiers_state[i]) {
218 32ff25bf bellard
            if (i & 0x80)
219 32ff25bf bellard
                kbd_put_keycode(0xe0);
220 32ff25bf bellard
            kbd_put_keycode(i | 0x80);
221 32ff25bf bellard
            modifiers_state[i] = 0;
222 32ff25bf bellard
        }
223 32ff25bf bellard
    }
224 32ff25bf bellard
}
225 32ff25bf bellard
226 0f0b7264 bellard
static void sdl_process_key(SDL_KeyboardEvent *ev)
227 0f0b7264 bellard
{
228 32ff25bf bellard
    int keycode, v;
229 de2200d3 bellard
230 de2200d3 bellard
    if (ev->keysym.sym == SDLK_PAUSE) {
231 de2200d3 bellard
        /* specific case */
232 de2200d3 bellard
        v = 0;
233 de2200d3 bellard
        if (ev->type == SDL_KEYUP)
234 de2200d3 bellard
            v |= 0x80;
235 de2200d3 bellard
        kbd_put_keycode(0xe1);
236 de2200d3 bellard
        kbd_put_keycode(0x1d | v);
237 de2200d3 bellard
        kbd_put_keycode(0x45 | v);
238 de2200d3 bellard
        return;
239 de2200d3 bellard
    }
240 de2200d3 bellard
241 3d11d0eb bellard
    if (kbd_layout) {
242 3d11d0eb bellard
        keycode = sdl_keyevent_to_keycode_generic(ev);
243 3d11d0eb bellard
    } else {
244 3d11d0eb bellard
        keycode = sdl_keyevent_to_keycode(ev);
245 3d11d0eb bellard
    }
246 de2200d3 bellard
247 de2200d3 bellard
    switch(keycode) {
248 de2200d3 bellard
    case 0x00:
249 de2200d3 bellard
        /* sent when leaving window: reset the modifiers state */
250 32ff25bf bellard
        reset_keys();
251 de2200d3 bellard
        return;
252 de2200d3 bellard
    case 0x2a:                          /* Left Shift */
253 de2200d3 bellard
    case 0x36:                          /* Right Shift */
254 de2200d3 bellard
    case 0x1d:                          /* Left CTRL */
255 de2200d3 bellard
    case 0x9d:                          /* Right CTRL */
256 de2200d3 bellard
    case 0x38:                          /* Left ALT */
257 de2200d3 bellard
    case 0xb8:                         /* Right ALT */
258 0f0b7264 bellard
        if (ev->type == SDL_KEYUP)
259 de2200d3 bellard
            modifiers_state[keycode] = 0;
260 de2200d3 bellard
        else
261 de2200d3 bellard
            modifiers_state[keycode] = 1;
262 de2200d3 bellard
        break;
263 de2200d3 bellard
    case 0x45: /* num lock */
264 de2200d3 bellard
    case 0x3a: /* caps lock */
265 de2200d3 bellard
        /* SDL does not send the key up event, so we generate it */
266 de2200d3 bellard
        kbd_put_keycode(keycode);
267 de2200d3 bellard
        kbd_put_keycode(keycode | 0x80);
268 de2200d3 bellard
        return;
269 0f0b7264 bellard
    }
270 de2200d3 bellard
271 de2200d3 bellard
    /* now send the key code */
272 de2200d3 bellard
    if (keycode & 0x80)
273 de2200d3 bellard
        kbd_put_keycode(0xe0);
274 de2200d3 bellard
    if (ev->type == SDL_KEYUP)
275 de2200d3 bellard
        kbd_put_keycode(keycode | 0x80);
276 de2200d3 bellard
    else
277 de2200d3 bellard
        kbd_put_keycode(keycode & 0x7f);
278 0f0b7264 bellard
}
279 0f0b7264 bellard
280 8a7ddc38 bellard
static void sdl_update_caption(void)
281 8a7ddc38 bellard
{
282 8a7ddc38 bellard
    char buf[1024];
283 c35734b2 ths
    const char *status = "";
284 c35734b2 ths
285 c35734b2 ths
    if (!vm_running)
286 c35734b2 ths
        status = " [Stopped]";
287 3780e197 ths
    else if (gui_grab) {
288 3780e197 ths
        if (!alt_grab)
289 3780e197 ths
            status = " - Press Ctrl-Alt to exit grab";
290 3780e197 ths
        else
291 3780e197 ths
            status = " - Press Ctrl-Alt-Shift to exit grab";
292 3780e197 ths
    }
293 c35734b2 ths
294 c35734b2 ths
    if (qemu_name)
295 c35734b2 ths
        snprintf(buf, sizeof(buf), "QEMU (%s)%s", qemu_name, status);
296 c35734b2 ths
    else
297 c35734b2 ths
        snprintf(buf, sizeof(buf), "QEMU%s", status);
298 c35734b2 ths
299 8a7ddc38 bellard
    SDL_WM_SetCaption(buf, "QEMU");
300 8a7ddc38 bellard
}
301 8a7ddc38 bellard
302 09b26c5e bellard
static void sdl_hide_cursor(void)
303 09b26c5e bellard
{
304 9467cd46 balrog
    if (!cursor_hide)
305 9467cd46 balrog
        return;
306 9467cd46 balrog
307 8785a8dd bellard
    if (kbd_mouse_is_absolute()) {
308 8785a8dd bellard
        SDL_ShowCursor(1);
309 8785a8dd bellard
        SDL_SetCursor(sdl_cursor_hidden);
310 8785a8dd bellard
    } else {
311 8785a8dd bellard
        SDL_ShowCursor(0);
312 8785a8dd bellard
    }
313 09b26c5e bellard
}
314 09b26c5e bellard
315 09b26c5e bellard
static void sdl_show_cursor(void)
316 09b26c5e bellard
{
317 9467cd46 balrog
    if (!cursor_hide)
318 9467cd46 balrog
        return;
319 9467cd46 balrog
320 09b26c5e bellard
    if (!kbd_mouse_is_absolute()) {
321 8785a8dd bellard
        SDL_ShowCursor(1);
322 d34cab9f ths
        if (guest_cursor &&
323 d34cab9f ths
                (gui_grab || kbd_mouse_is_absolute() || absolute_enabled))
324 d34cab9f ths
            SDL_SetCursor(guest_sprite);
325 d34cab9f ths
        else
326 d34cab9f ths
            SDL_SetCursor(sdl_cursor_normal);
327 09b26c5e bellard
    }
328 09b26c5e bellard
}
329 09b26c5e bellard
330 0f0b7264 bellard
static void sdl_grab_start(void)
331 0f0b7264 bellard
{
332 d34cab9f ths
    if (guest_cursor) {
333 d34cab9f ths
        SDL_SetCursor(guest_sprite);
334 08a2d4c4 balrog
        if (!kbd_mouse_is_absolute() && !absolute_enabled)
335 08a2d4c4 balrog
            SDL_WarpMouse(guest_x, guest_y);
336 d34cab9f ths
    } else
337 d34cab9f ths
        sdl_hide_cursor();
338 6bb81603 aliguori
339 6bb81603 aliguori
    if (SDL_WM_GrabInput(SDL_GRAB_ON) == SDL_GRAB_ON) {
340 6bb81603 aliguori
        gui_grab = 1;
341 6bb81603 aliguori
        sdl_update_caption();
342 6bb81603 aliguori
    } else
343 6bb81603 aliguori
        sdl_show_cursor();
344 0f0b7264 bellard
}
345 0f0b7264 bellard
346 0f0b7264 bellard
static void sdl_grab_end(void)
347 0f0b7264 bellard
{
348 0f0b7264 bellard
    SDL_WM_GrabInput(SDL_GRAB_OFF);
349 0f0b7264 bellard
    gui_grab = 0;
350 d34cab9f ths
    sdl_show_cursor();
351 8a7ddc38 bellard
    sdl_update_caption();
352 0f0b7264 bellard
}
353 0f0b7264 bellard
354 4c44bdcb aurel32
static void sdl_send_mouse_event(int dx, int dy, int dz, int x, int y, int state)
355 0f0b7264 bellard
{
356 4c44bdcb aurel32
    int buttons;
357 0f0b7264 bellard
    buttons = 0;
358 0f0b7264 bellard
    if (state & SDL_BUTTON(SDL_BUTTON_LEFT))
359 0f0b7264 bellard
        buttons |= MOUSE_EVENT_LBUTTON;
360 0f0b7264 bellard
    if (state & SDL_BUTTON(SDL_BUTTON_RIGHT))
361 0f0b7264 bellard
        buttons |= MOUSE_EVENT_RBUTTON;
362 0f0b7264 bellard
    if (state & SDL_BUTTON(SDL_BUTTON_MIDDLE))
363 0f0b7264 bellard
        buttons |= MOUSE_EVENT_MBUTTON;
364 09b26c5e bellard
365 09b26c5e bellard
    if (kbd_mouse_is_absolute()) {
366 09b26c5e bellard
        if (!absolute_enabled) {
367 09b26c5e bellard
            sdl_hide_cursor();
368 09b26c5e bellard
            if (gui_grab) {
369 09b26c5e bellard
                sdl_grab_end();
370 09b26c5e bellard
            }
371 09b26c5e bellard
            absolute_enabled = 1;
372 09b26c5e bellard
        }
373 09b26c5e bellard
374 4c44bdcb aurel32
       dx = x * 0x7FFF / (width - 1);
375 4c44bdcb aurel32
       dy = y * 0x7FFF / (height - 1);
376 455204eb ths
    } else if (absolute_enabled) {
377 455204eb ths
        sdl_show_cursor();
378 455204eb ths
        absolute_enabled = 0;
379 d34cab9f ths
    } else if (guest_cursor) {
380 4c44bdcb aurel32
        x -= guest_x;
381 4c44bdcb aurel32
        y -= guest_y;
382 4c44bdcb aurel32
        guest_x += x;
383 4c44bdcb aurel32
        guest_y += y;
384 4c44bdcb aurel32
        dx = x;
385 4c44bdcb aurel32
        dy = y;
386 09b26c5e bellard
    }
387 09b26c5e bellard
388 0f0b7264 bellard
    kbd_mouse_event(dx, dy, dz, buttons);
389 0f0b7264 bellard
}
390 0f0b7264 bellard
391 8e9c4afe bellard
static void toggle_full_screen(DisplayState *ds)
392 8e9c4afe bellard
{
393 8e9c4afe bellard
    gui_fullscreen = !gui_fullscreen;
394 7d957bd8 aliguori
    sdl_resize(ds);
395 8e9c4afe bellard
    if (gui_fullscreen) {
396 8e9c4afe bellard
        gui_saved_grab = gui_grab;
397 8e9c4afe bellard
        sdl_grab_start();
398 8e9c4afe bellard
    } else {
399 8e9c4afe bellard
        if (!gui_saved_grab)
400 8e9c4afe bellard
            sdl_grab_end();
401 8e9c4afe bellard
    }
402 95219897 pbrook
    vga_hw_invalidate();
403 95219897 pbrook
    vga_hw_update();
404 8e9c4afe bellard
}
405 8e9c4afe bellard
406 0f0b7264 bellard
static void sdl_refresh(DisplayState *ds)
407 0f0b7264 bellard
{
408 0f0b7264 bellard
    SDL_Event ev1, *ev = &ev1;
409 8e9c4afe bellard
    int mod_state;
410 4c44bdcb aurel32
    int buttonstate = SDL_GetMouseState(NULL, NULL);
411 3b46e624 ths
412 8a7ddc38 bellard
    if (last_vm_running != vm_running) {
413 8a7ddc38 bellard
        last_vm_running = vm_running;
414 8a7ddc38 bellard
        sdl_update_caption();
415 8a7ddc38 bellard
    }
416 8a7ddc38 bellard
417 95219897 pbrook
    vga_hw_update();
418 3bee8bd0 aurel32
    SDL_EnableUNICODE(!is_graphic_console());
419 457831f4 bellard
420 0f0b7264 bellard
    while (SDL_PollEvent(ev)) {
421 0f0b7264 bellard
        switch (ev->type) {
422 0f0b7264 bellard
        case SDL_VIDEOEXPOSE:
423 7d957bd8 aliguori
            sdl_update(ds, 0, 0, real_screen->w, real_screen->h);
424 0f0b7264 bellard
            break;
425 0f0b7264 bellard
        case SDL_KEYDOWN:
426 0f0b7264 bellard
        case SDL_KEYUP:
427 0f0b7264 bellard
            if (ev->type == SDL_KEYDOWN) {
428 3780e197 ths
                if (!alt_grab) {
429 3780e197 ths
                    mod_state = (SDL_GetModState() & gui_grab_code) ==
430 3780e197 ths
                                gui_grab_code;
431 3780e197 ths
                } else {
432 3780e197 ths
                    mod_state = (SDL_GetModState() & (gui_grab_code | KMOD_LSHIFT)) ==
433 3780e197 ths
                                (gui_grab_code | KMOD_LSHIFT);
434 3780e197 ths
                }
435 8e9c4afe bellard
                gui_key_modifier_pressed = mod_state;
436 457831f4 bellard
                if (gui_key_modifier_pressed) {
437 32ff25bf bellard
                    int keycode;
438 32ff25bf bellard
                    keycode = sdl_keyevent_to_keycode(&ev->key);
439 32ff25bf bellard
                    switch(keycode) {
440 32ff25bf bellard
                    case 0x21: /* 'f' key on US keyboard */
441 457831f4 bellard
                        toggle_full_screen(ds);
442 457831f4 bellard
                        gui_keysym = 1;
443 457831f4 bellard
                        break;
444 5fafdf24 ths
                    case 0x02 ... 0x0a: /* '1' to '9' keys */
445 dfd92d3a bellard
                        /* Reset the modifiers sent to the current console */
446 dfd92d3a bellard
                        reset_keys();
447 32ff25bf bellard
                        console_select(keycode - 0x02);
448 95219897 pbrook
                        if (!is_graphic_console()) {
449 457831f4 bellard
                            /* display grab if going to a text console */
450 457831f4 bellard
                            if (gui_grab)
451 457831f4 bellard
                                sdl_grab_end();
452 457831f4 bellard
                        }
453 457831f4 bellard
                        gui_keysym = 1;
454 457831f4 bellard
                        break;
455 457831f4 bellard
                    default:
456 457831f4 bellard
                        break;
457 457831f4 bellard
                    }
458 95219897 pbrook
                } else if (!is_graphic_console()) {
459 457831f4 bellard
                    int keysym;
460 457831f4 bellard
                    keysym = 0;
461 457831f4 bellard
                    if (ev->key.keysym.mod & (KMOD_LCTRL | KMOD_RCTRL)) {
462 457831f4 bellard
                        switch(ev->key.keysym.sym) {
463 457831f4 bellard
                        case SDLK_UP: keysym = QEMU_KEY_CTRL_UP; break;
464 457831f4 bellard
                        case SDLK_DOWN: keysym = QEMU_KEY_CTRL_DOWN; break;
465 457831f4 bellard
                        case SDLK_LEFT: keysym = QEMU_KEY_CTRL_LEFT; break;
466 457831f4 bellard
                        case SDLK_RIGHT: keysym = QEMU_KEY_CTRL_RIGHT; break;
467 457831f4 bellard
                        case SDLK_HOME: keysym = QEMU_KEY_CTRL_HOME; break;
468 457831f4 bellard
                        case SDLK_END: keysym = QEMU_KEY_CTRL_END; break;
469 457831f4 bellard
                        case SDLK_PAGEUP: keysym = QEMU_KEY_CTRL_PAGEUP; break;
470 457831f4 bellard
                        case SDLK_PAGEDOWN: keysym = QEMU_KEY_CTRL_PAGEDOWN; break;
471 457831f4 bellard
                        default: break;
472 457831f4 bellard
                        }
473 457831f4 bellard
                    } else {
474 457831f4 bellard
                        switch(ev->key.keysym.sym) {
475 457831f4 bellard
                        case SDLK_UP: keysym = QEMU_KEY_UP; break;
476 457831f4 bellard
                        case SDLK_DOWN: keysym = QEMU_KEY_DOWN; break;
477 457831f4 bellard
                        case SDLK_LEFT: keysym = QEMU_KEY_LEFT; break;
478 457831f4 bellard
                        case SDLK_RIGHT: keysym = QEMU_KEY_RIGHT; break;
479 457831f4 bellard
                        case SDLK_HOME: keysym = QEMU_KEY_HOME; break;
480 457831f4 bellard
                        case SDLK_END: keysym = QEMU_KEY_END; break;
481 457831f4 bellard
                        case SDLK_PAGEUP: keysym = QEMU_KEY_PAGEUP; break;
482 457831f4 bellard
                        case SDLK_PAGEDOWN: keysym = QEMU_KEY_PAGEDOWN; break;
483 e91c8a77 ths
                        case SDLK_BACKSPACE: keysym = QEMU_KEY_BACKSPACE; break;
484 e91c8a77 ths
                        case SDLK_DELETE: keysym = QEMU_KEY_DELETE; break;
485 457831f4 bellard
                        default: break;
486 457831f4 bellard
                        }
487 457831f4 bellard
                    }
488 457831f4 bellard
                    if (keysym) {
489 457831f4 bellard
                        kbd_put_keysym(keysym);
490 457831f4 bellard
                    } else if (ev->key.keysym.unicode != 0) {
491 457831f4 bellard
                        kbd_put_keysym(ev->key.keysym.unicode);
492 457831f4 bellard
                    }
493 8e9c4afe bellard
                }
494 8e9c4afe bellard
            } else if (ev->type == SDL_KEYUP) {
495 3780e197 ths
                if (!alt_grab) {
496 3780e197 ths
                    mod_state = (ev->key.keysym.mod & gui_grab_code);
497 3780e197 ths
                } else {
498 3780e197 ths
                    mod_state = (ev->key.keysym.mod &
499 3780e197 ths
                                 (gui_grab_code | KMOD_LSHIFT));
500 3780e197 ths
                }
501 8e9c4afe bellard
                if (!mod_state) {
502 8e9c4afe bellard
                    if (gui_key_modifier_pressed) {
503 5b311878 pbrook
                        gui_key_modifier_pressed = 0;
504 457831f4 bellard
                        if (gui_keysym == 0) {
505 32ff25bf bellard
                            /* exit/enter grab if pressing Ctrl-Alt */
506 c66b0d4c bellard
                            if (!gui_grab) {
507 c66b0d4c bellard
                                /* if the application is not active,
508 c66b0d4c bellard
                                   do not try to enter grab state. It
509 c66b0d4c bellard
                                   prevents
510 c66b0d4c bellard
                                   'SDL_WM_GrabInput(SDL_GRAB_ON)'
511 c66b0d4c bellard
                                   from blocking all the application
512 c66b0d4c bellard
                                   (SDL bug). */
513 c66b0d4c bellard
                                if (SDL_GetAppState() & SDL_APPACTIVE)
514 c66b0d4c bellard
                                    sdl_grab_start();
515 c66b0d4c bellard
                            } else {
516 8e9c4afe bellard
                                sdl_grab_end();
517 c66b0d4c bellard
                            }
518 32ff25bf bellard
                            /* SDL does not send back all the
519 32ff25bf bellard
                               modifiers key, so we must correct it */
520 32ff25bf bellard
                            reset_keys();
521 8e9c4afe bellard
                            break;
522 8e9c4afe bellard
                        }
523 8e9c4afe bellard
                        gui_keysym = 0;
524 8e9c4afe bellard
                    }
525 0f0b7264 bellard
                }
526 0f0b7264 bellard
            }
527 5fafdf24 ths
            if (is_graphic_console() && !gui_keysym)
528 457831f4 bellard
                sdl_process_key(&ev->key);
529 0f0b7264 bellard
            break;
530 0f0b7264 bellard
        case SDL_QUIT:
531 5b08fc10 aliguori
            if (!no_quit)
532 731345e1 balrog
                qemu_system_shutdown_request();
533 0f0b7264 bellard
            break;
534 0f0b7264 bellard
        case SDL_MOUSEMOTION:
535 455204eb ths
            if (gui_grab || kbd_mouse_is_absolute() ||
536 455204eb ths
                absolute_enabled) {
537 4c44bdcb aurel32
                sdl_send_mouse_event(ev->motion.xrel, ev->motion.yrel, 0,
538 4c44bdcb aurel32
                       ev->motion.x, ev->motion.y, ev->motion.state);
539 0f0b7264 bellard
            }
540 0f0b7264 bellard
            break;
541 0f0b7264 bellard
        case SDL_MOUSEBUTTONDOWN:
542 0f0b7264 bellard
        case SDL_MOUSEBUTTONUP:
543 0f0b7264 bellard
            {
544 0f0b7264 bellard
                SDL_MouseButtonEvent *bev = &ev->button;
545 09b26c5e bellard
                if (!gui_grab && !kbd_mouse_is_absolute()) {
546 0f0b7264 bellard
                    if (ev->type == SDL_MOUSEBUTTONDOWN &&
547 4c44bdcb aurel32
                        (bev->button == SDL_BUTTON_LEFT)) {
548 0f0b7264 bellard
                        /* start grabbing all events */
549 0f0b7264 bellard
                        sdl_grab_start();
550 0f0b7264 bellard
                    }
551 0f0b7264 bellard
                } else {
552 18a6d284 bellard
                    int dz;
553 18a6d284 bellard
                    dz = 0;
554 4c44bdcb aurel32
                    if (ev->type == SDL_MOUSEBUTTONDOWN) {
555 4c44bdcb aurel32
                        buttonstate |= SDL_BUTTON(bev->button);
556 4c44bdcb aurel32
                    } else {
557 4c44bdcb aurel32
                        buttonstate &= ~SDL_BUTTON(bev->button);
558 4c44bdcb aurel32
                    }
559 18a6d284 bellard
#ifdef SDL_BUTTON_WHEELUP
560 09b26c5e bellard
                    if (bev->button == SDL_BUTTON_WHEELUP && ev->type == SDL_MOUSEBUTTONDOWN) {
561 18a6d284 bellard
                        dz = -1;
562 09b26c5e bellard
                    } else if (bev->button == SDL_BUTTON_WHEELDOWN && ev->type == SDL_MOUSEBUTTONDOWN) {
563 18a6d284 bellard
                        dz = 1;
564 18a6d284 bellard
                    }
565 3b46e624 ths
#endif
566 4c44bdcb aurel32
                    sdl_send_mouse_event(0, 0, dz, bev->x, bev->y, buttonstate);
567 0f0b7264 bellard
                }
568 0f0b7264 bellard
            }
569 0f0b7264 bellard
            break;
570 0294ffb9 bellard
        case SDL_ACTIVEEVENT:
571 5b311878 pbrook
            if (gui_grab && ev->active.state == SDL_APPINPUTFOCUS &&
572 5b311878 pbrook
                !ev->active.gain && !gui_fullscreen_initial_grab) {
573 0294ffb9 bellard
                sdl_grab_end();
574 0294ffb9 bellard
            }
575 f442e08b aurel32
            if (ev->active.state & SDL_APPACTIVE) {
576 f442e08b aurel32
                if (ev->active.gain) {
577 f442e08b aurel32
                    /* Back to default interval */
578 7d957bd8 aliguori
                    dcl->gui_timer_interval = 0;
579 7d957bd8 aliguori
                    dcl->idle = 0;
580 f442e08b aurel32
                } else {
581 f442e08b aurel32
                    /* Sleeping interval */
582 7d957bd8 aliguori
                    dcl->gui_timer_interval = 500;
583 7d957bd8 aliguori
                    dcl->idle = 1;
584 f442e08b aurel32
                }
585 f442e08b aurel32
            }
586 0294ffb9 bellard
            break;
587 0f0b7264 bellard
        default:
588 0f0b7264 bellard
            break;
589 0f0b7264 bellard
        }
590 0f0b7264 bellard
    }
591 0f0b7264 bellard
}
592 0f0b7264 bellard
593 d34cab9f ths
static void sdl_fill(DisplayState *ds, int x, int y, int w, int h, uint32_t c)
594 d34cab9f ths
{
595 d34cab9f ths
    SDL_Rect dst = { x, y, w, h };
596 7d957bd8 aliguori
    SDL_FillRect(real_screen, &dst, c);
597 d34cab9f ths
}
598 d34cab9f ths
599 d34cab9f ths
static void sdl_mouse_warp(int x, int y, int on)
600 d34cab9f ths
{
601 d34cab9f ths
    if (on) {
602 d34cab9f ths
        if (!guest_cursor)
603 d34cab9f ths
            sdl_show_cursor();
604 d34cab9f ths
        if (gui_grab || kbd_mouse_is_absolute() || absolute_enabled) {
605 d34cab9f ths
            SDL_SetCursor(guest_sprite);
606 08a2d4c4 balrog
            if (!kbd_mouse_is_absolute() && !absolute_enabled)
607 08a2d4c4 balrog
                SDL_WarpMouse(x, y);
608 d34cab9f ths
        }
609 d34cab9f ths
    } else if (gui_grab)
610 d34cab9f ths
        sdl_hide_cursor();
611 d34cab9f ths
    guest_cursor = on;
612 d34cab9f ths
    guest_x = x, guest_y = y;
613 d34cab9f ths
}
614 d34cab9f ths
615 d34cab9f ths
static void sdl_mouse_define(int width, int height, int bpp,
616 d34cab9f ths
                             int hot_x, int hot_y,
617 d34cab9f ths
                             uint8_t *image, uint8_t *mask)
618 d34cab9f ths
{
619 d34cab9f ths
    uint8_t sprite[256], *line;
620 d34cab9f ths
    int x, y, dst, bypl, src = 0;
621 d34cab9f ths
    if (guest_sprite)
622 d34cab9f ths
        SDL_FreeCursor(guest_sprite);
623 d34cab9f ths
624 d34cab9f ths
    memset(sprite, 0, 256);
625 d34cab9f ths
    bypl = ((width * bpp + 31) >> 5) << 2;
626 d34cab9f ths
    for (y = 0, dst = 0; y < height; y ++, image += bypl) {
627 d34cab9f ths
        line = image;
628 d34cab9f ths
        for (x = 0; x < width; x ++, dst ++) {
629 d34cab9f ths
            switch (bpp) {
630 d34cab9f ths
            case 24:
631 d34cab9f ths
                src = *(line ++); src |= *(line ++); src |= *(line ++);
632 d34cab9f ths
                break;
633 d34cab9f ths
            case 16:
634 d34cab9f ths
            case 15:
635 d34cab9f ths
                src = *(line ++); src |= *(line ++);
636 d34cab9f ths
                break;
637 d34cab9f ths
            case 8:
638 d34cab9f ths
                src = *(line ++);
639 d34cab9f ths
                break;
640 d34cab9f ths
            case 4:
641 d34cab9f ths
                src = 0xf & (line[x >> 1] >> ((x & 1)) << 2);
642 d34cab9f ths
                break;
643 d34cab9f ths
            case 2:
644 d34cab9f ths
                src = 3 & (line[x >> 2] >> ((x & 3)) << 1);
645 d34cab9f ths
                break;
646 d34cab9f ths
            case 1:
647 d34cab9f ths
                src = 1 & (line[x >> 3] >> (x & 7));
648 d34cab9f ths
                break;
649 d34cab9f ths
            }
650 d34cab9f ths
            if (!src)
651 d34cab9f ths
                sprite[dst >> 3] |= (1 << (~dst & 7)) & mask[dst >> 3];
652 d34cab9f ths
        }
653 d34cab9f ths
    }
654 d34cab9f ths
    guest_sprite = SDL_CreateCursor(sprite, mask, width, height, hot_x, hot_y);
655 d34cab9f ths
656 d34cab9f ths
    if (guest_cursor &&
657 d34cab9f ths
            (gui_grab || kbd_mouse_is_absolute() || absolute_enabled))
658 d34cab9f ths
        SDL_SetCursor(guest_sprite);
659 d34cab9f ths
}
660 d34cab9f ths
661 5fafdf24 ths
static void sdl_cleanup(void)
662 898712a8 bellard
{
663 d34cab9f ths
    if (guest_sprite)
664 d34cab9f ths
        SDL_FreeCursor(guest_sprite);
665 898712a8 bellard
    SDL_Quit();
666 898712a8 bellard
}
667 898712a8 bellard
668 43523e93 ths
void sdl_display_init(DisplayState *ds, int full_screen, int no_frame)
669 0f0b7264 bellard
{
670 0f0b7264 bellard
    int flags;
671 09b26c5e bellard
    uint8_t data = 0;
672 0f0b7264 bellard
673 3d11d0eb bellard
#if defined(__APPLE__)
674 3d11d0eb bellard
    /* always use generic keymaps */
675 3d11d0eb bellard
    if (!keyboard_layout)
676 3d11d0eb bellard
        keyboard_layout = "en-us";
677 3d11d0eb bellard
#endif
678 3d11d0eb bellard
    if(keyboard_layout) {
679 0483755a aliguori
        kbd_layout = init_keyboard_layout(name2keysym, keyboard_layout);
680 3d11d0eb bellard
        if (!kbd_layout)
681 3d11d0eb bellard
            exit(1);
682 3d11d0eb bellard
    }
683 3d11d0eb bellard
684 43523e93 ths
    if (no_frame)
685 43523e93 ths
        gui_noframe = 1;
686 43523e93 ths
687 0f0b7264 bellard
    flags = SDL_INIT_VIDEO | SDL_INIT_NOPARACHUTE;
688 0f0b7264 bellard
    if (SDL_Init (flags)) {
689 0f0b7264 bellard
        fprintf(stderr, "Could not initialize SDL - exiting\n");
690 0f0b7264 bellard
        exit(1);
691 0f0b7264 bellard
    }
692 0ae04d73 bellard
693 7d957bd8 aliguori
    dcl = qemu_mallocz(sizeof(DisplayChangeListener));
694 7d957bd8 aliguori
    dcl->dpy_update = sdl_update;
695 7d957bd8 aliguori
    dcl->dpy_resize = sdl_resize;
696 7d957bd8 aliguori
    dcl->dpy_refresh = sdl_refresh;
697 7d957bd8 aliguori
    dcl->dpy_setdata = sdl_setdata;
698 7d957bd8 aliguori
    dcl->dpy_fill = sdl_fill;
699 d34cab9f ths
    ds->mouse_set = sdl_mouse_warp;
700 d34cab9f ths
    ds->cursor_define = sdl_mouse_define;
701 7d957bd8 aliguori
    register_displaychangelistener(ds, dcl);
702 0f0b7264 bellard
703 8a7ddc38 bellard
    sdl_update_caption();
704 0f0b7264 bellard
    SDL_EnableKeyRepeat(250, 50);
705 0f0b7264 bellard
    gui_grab = 0;
706 898712a8 bellard
707 09b26c5e bellard
    sdl_cursor_hidden = SDL_CreateCursor(&data, &data, 8, 1, 0, 0);
708 09b26c5e bellard
    sdl_cursor_normal = SDL_GetCursor();
709 09b26c5e bellard
710 898712a8 bellard
    atexit(sdl_cleanup);
711 d63d307f bellard
    if (full_screen) {
712 d63d307f bellard
        gui_fullscreen = 1;
713 d63d307f bellard
        gui_fullscreen_initial_grab = 1;
714 d63d307f bellard
        sdl_grab_start();
715 d63d307f bellard
    }
716 0f0b7264 bellard
}