Statistics
| Branch: | Revision:

root / ui / curses.c @ cd100328

History | View | Annotate | Download (9.9 kB)

1
/*
2
 * QEMU curses/ncurses display driver
3
 * 
4
 * Copyright (c) 2005 Andrzej Zaborowski  <balrog@zabor.org>
5
 * 
6
 * Permission is hereby granted, free of charge, to any person obtaining a copy
7
 * of this software and associated documentation files (the "Software"), to deal
8
 * in the Software without restriction, including without limitation the rights
9
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10
 * copies of the Software, and to permit persons to whom the Software is
11
 * furnished to do so, subject to the following conditions:
12
 *
13
 * The above copyright notice and this permission notice shall be included in
14
 * all copies or substantial portions of the Software.
15
 *
16
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
19
 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22
 * THE SOFTWARE.
23
 */
24
#include <curses.h>
25

    
26
#ifndef _WIN32
27
#include <sys/ioctl.h>
28
#include <termios.h>
29
#endif
30

    
31
#include "qemu-common.h"
32
#include "ui/console.h"
33
#include "ui/input.h"
34
#include "sysemu/sysemu.h"
35

    
36
#define FONT_HEIGHT 16
37
#define FONT_WIDTH 8
38

    
39
static DisplayChangeListener *dcl;
40
static console_ch_t screen[160 * 100];
41
static WINDOW *screenpad = NULL;
42
static int width, height, gwidth, gheight, invalidate;
43
static int px, py, sminx, sminy, smaxx, smaxy;
44

    
45
static void curses_update(DisplayChangeListener *dcl,
46
                          int x, int y, int w, int h)
47
{
48
    chtype *line;
49

    
50
    line = ((chtype *) screen) + y * width;
51
    for (h += y; y < h; y ++, line += width)
52
        mvwaddchnstr(screenpad, y, 0, line, width);
53

    
54
    pnoutrefresh(screenpad, py, px, sminy, sminx, smaxy - 1, smaxx - 1);
55
    refresh();
56
}
57

    
58
static void curses_calc_pad(void)
59
{
60
    if (qemu_console_is_fixedsize(NULL)) {
61
        width = gwidth;
62
        height = gheight;
63
    } else {
64
        width = COLS;
65
        height = LINES;
66
    }
67

    
68
    if (screenpad)
69
        delwin(screenpad);
70

    
71
    clear();
72
    refresh();
73

    
74
    screenpad = newpad(height, width);
75

    
76
    if (width > COLS) {
77
        px = (width - COLS) / 2;
78
        sminx = 0;
79
        smaxx = COLS;
80
    } else {
81
        px = 0;
82
        sminx = (COLS - width) / 2;
83
        smaxx = sminx + width;
84
    }
85

    
86
    if (height > LINES) {
87
        py = (height - LINES) / 2;
88
        sminy = 0;
89
        smaxy = LINES;
90
    } else {
91
        py = 0;
92
        sminy = (LINES - height) / 2;
93
        smaxy = sminy + height;
94
    }
95
}
96

    
97
static void curses_resize(DisplayChangeListener *dcl,
98
                          int width, int height)
99
{
100
    if (width == gwidth && height == gheight) {
101
        return;
102
    }
103

    
104
    gwidth = width;
105
    gheight = height;
106

    
107
    curses_calc_pad();
108
}
109

    
110
#if !defined(_WIN32) && defined(SIGWINCH) && defined(KEY_RESIZE)
111
static volatile sig_atomic_t got_sigwinch;
112
static void curses_winch_check(void)
113
{
114
    struct winsize {
115
        unsigned short ws_row;
116
        unsigned short ws_col;
117
        unsigned short ws_xpixel;   /* unused */
118
        unsigned short ws_ypixel;   /* unused */
119
    } ws;
120

    
121
    if (!got_sigwinch) {
122
        return;
123
    }
124
    got_sigwinch = false;
125

    
126
    if (ioctl(1, TIOCGWINSZ, &ws) == -1) {
127
        return;
128
    }
129

    
130
    resize_term(ws.ws_row, ws.ws_col);
131
    invalidate = 1;
132
}
133

    
134
static void curses_winch_handler(int signum)
135
{
136
    got_sigwinch = true;
137
}
138

    
139
static void curses_winch_init(void)
140
{
141
    struct sigaction old, winch = {
142
        .sa_handler  = curses_winch_handler,
143
    };
144
    sigaction(SIGWINCH, &winch, &old);
145
}
146
#else
147
static void curses_winch_check(void) {}
148
static void curses_winch_init(void) {}
149
#endif
150

    
151
static void curses_cursor_position(DisplayChangeListener *dcl,
152
                                   int x, int y)
153
{
154
    if (x >= 0) {
155
        x = sminx + x - px;
156
        y = sminy + y - py;
157

    
158
        if (x >= 0 && y >= 0 && x < COLS && y < LINES) {
159
            move(y, x);
160
            curs_set(1);
161
            /* it seems that curs_set(1) must always be called before
162
             * curs_set(2) for the latter to have effect */
163
            if (!qemu_console_is_graphic(NULL)) {
164
                curs_set(2);
165
            }
166
            return;
167
        }
168
    }
169

    
170
    curs_set(0);
171
}
172

    
173
/* generic keyboard conversion */
174

    
175
#include "curses_keys.h"
176

    
177
static kbd_layout_t *kbd_layout = NULL;
178

    
179
static void curses_refresh(DisplayChangeListener *dcl)
180
{
181
    int chr, nextchr, keysym, keycode, keycode_alt;
182

    
183
    curses_winch_check();
184

    
185
    if (invalidate) {
186
        clear();
187
        refresh();
188
        curses_calc_pad();
189
        graphic_hw_invalidate(NULL);
190
        invalidate = 0;
191
    }
192

    
193
    graphic_hw_text_update(NULL, screen);
194

    
195
    nextchr = ERR;
196
    while (1) {
197
        /* while there are any pending key strokes to process */
198
        if (nextchr == ERR)
199
            chr = getch();
200
        else {
201
            chr = nextchr;
202
            nextchr = ERR;
203
        }
204

    
205
        if (chr == ERR)
206
            break;
207

    
208
#ifdef KEY_RESIZE
209
        /* this shouldn't occur when we use a custom SIGWINCH handler */
210
        if (chr == KEY_RESIZE) {
211
            clear();
212
            refresh();
213
            curses_calc_pad();
214
            curses_update(dcl, 0, 0, width, height);
215
            continue;
216
        }
217
#endif
218

    
219
        keycode = curses2keycode[chr];
220
        keycode_alt = 0;
221

    
222
        /* alt key */
223
        if (keycode == 1) {
224
            nextchr = getch();
225

    
226
            if (nextchr != ERR) {
227
                chr = nextchr;
228
                keycode_alt = ALT;
229
                keycode = curses2keycode[nextchr];
230
                nextchr = ERR;
231

    
232
                if (keycode != -1) {
233
                    keycode |= ALT;
234

    
235
                    /* process keys reserved for qemu */
236
                    if (keycode >= QEMU_KEY_CONSOLE0 &&
237
                            keycode < QEMU_KEY_CONSOLE0 + 9) {
238
                        erase();
239
                        wnoutrefresh(stdscr);
240
                        console_select(keycode - QEMU_KEY_CONSOLE0);
241

    
242
                        invalidate = 1;
243
                        continue;
244
                    }
245
                }
246
            }
247
        }
248

    
249
        if (kbd_layout) {
250
            keysym = -1;
251
            if (chr < CURSES_KEYS)
252
                keysym = curses2keysym[chr];
253

    
254
            if (keysym == -1) {
255
                if (chr < ' ') {
256
                    keysym = chr + '@';
257
                    if (keysym >= 'A' && keysym <= 'Z')
258
                        keysym += 'a' - 'A';
259
                    keysym |= KEYSYM_CNTRL;
260
                } else
261
                    keysym = chr;
262
            }
263

    
264
            keycode = keysym2scancode(kbd_layout, keysym & KEYSYM_MASK);
265
            if (keycode == 0)
266
                continue;
267

    
268
            keycode |= (keysym & ~KEYSYM_MASK) >> 16;
269
            keycode |= keycode_alt;
270
        }
271

    
272
        if (keycode == -1)
273
            continue;
274

    
275
        if (qemu_console_is_graphic(NULL)) {
276
            /* since terminals don't know about key press and release
277
             * events, we need to emit both for each key received */
278
            if (keycode & SHIFT) {
279
                qemu_input_event_send_key_number(NULL, SHIFT_CODE, true);
280
            }
281
            if (keycode & CNTRL) {
282
                qemu_input_event_send_key_number(NULL, CNTRL_CODE, true);
283
            }
284
            if (keycode & ALT) {
285
                qemu_input_event_send_key_number(NULL, ALT_CODE, true);
286
            }
287
            if (keycode & ALTGR) {
288
                qemu_input_event_send_key_number(NULL, GREY | ALT_CODE, true);
289
            }
290

    
291
            qemu_input_event_send_key_number(NULL, keycode, true);
292
            qemu_input_event_send_key_number(NULL, keycode, false);
293

    
294
            if (keycode & ALTGR) {
295
                qemu_input_event_send_key_number(NULL, GREY | ALT_CODE, false);
296
            }
297
            if (keycode & ALT) {
298
                qemu_input_event_send_key_number(NULL, ALT_CODE, false);
299
            }
300
            if (keycode & CNTRL) {
301
                qemu_input_event_send_key_number(NULL, CNTRL_CODE, false);
302
            }
303
            if (keycode & SHIFT) {
304
                qemu_input_event_send_key_number(NULL, SHIFT_CODE, false);
305
            }
306
        } else {
307
            keysym = curses2qemu[chr];
308
            if (keysym == -1)
309
                keysym = chr;
310

    
311
            kbd_put_keysym(keysym);
312
        }
313
    }
314
}
315

    
316
static void curses_atexit(void)
317
{
318
    endwin();
319
}
320

    
321
static void curses_setup(void)
322
{
323
    int i, colour_default[8] = {
324
        COLOR_BLACK, COLOR_BLUE, COLOR_GREEN, COLOR_CYAN,
325
        COLOR_RED, COLOR_MAGENTA, COLOR_YELLOW, COLOR_WHITE,
326
    };
327

    
328
    /* input as raw as possible, let everything be interpreted
329
     * by the guest system */
330
    initscr(); noecho(); intrflush(stdscr, FALSE);
331
    nodelay(stdscr, TRUE); nonl(); keypad(stdscr, TRUE);
332
    start_color(); raw(); scrollok(stdscr, FALSE);
333

    
334
    for (i = 0; i < 64; i ++)
335
        init_pair(i, colour_default[i & 7], colour_default[i >> 3]);
336
}
337

    
338
static void curses_keyboard_setup(void)
339
{
340
#if defined(__APPLE__)
341
    /* always use generic keymaps */
342
    if (!keyboard_layout)
343
        keyboard_layout = "en-us";
344
#endif
345
    if(keyboard_layout) {
346
        kbd_layout = init_keyboard_layout(name2keysym, keyboard_layout);
347
        if (!kbd_layout)
348
            exit(1);
349
    }
350
}
351

    
352
static const DisplayChangeListenerOps dcl_ops = {
353
    .dpy_name        = "curses",
354
    .dpy_text_update = curses_update,
355
    .dpy_text_resize = curses_resize,
356
    .dpy_refresh     = curses_refresh,
357
    .dpy_text_cursor = curses_cursor_position,
358
};
359

    
360
void curses_display_init(DisplayState *ds, int full_screen)
361
{
362
#ifndef _WIN32
363
    if (!isatty(1)) {
364
        fprintf(stderr, "We need a terminal output\n");
365
        exit(1);
366
    }
367
#endif
368

    
369
    curses_setup();
370
    curses_keyboard_setup();
371
    atexit(curses_atexit);
372

    
373
    curses_winch_init();
374

    
375
    dcl = (DisplayChangeListener *) g_malloc0(sizeof(DisplayChangeListener));
376
    dcl->ops = &dcl_ops;
377
    register_displaychangelistener(dcl);
378

    
379
    invalidate = 1;
380
}