2004-07-14 17:28:59 +00:00
|
|
|
/*
|
|
|
|
* QEMU graphical console
|
2007-09-16 21:08:06 +00:00
|
|
|
*
|
2004-07-14 17:28:59 +00:00
|
|
|
* Copyright (c) 2004 Fabrice Bellard
|
2007-09-16 21:08:06 +00:00
|
|
|
*
|
2004-07-14 17:28:59 +00:00
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
|
|
* in the Software without restriction, including without limitation the rights
|
|
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
*
|
|
|
|
* The above copyright notice and this permission notice shall be included in
|
|
|
|
* all copies or substantial portions of the Software.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
|
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
|
|
* THE SOFTWARE.
|
|
|
|
*/
|
2007-11-17 17:14:51 +00:00
|
|
|
#include "qemu-common.h"
|
2012-11-28 12:06:30 +01:00
|
|
|
#include "ui/console.h"
|
2012-12-17 18:20:00 +01:00
|
|
|
#include "qemu/timer.h"
|
2012-05-24 13:48:23 -03:00
|
|
|
#include "qmp-commands.h"
|
2012-12-17 18:20:05 +01:00
|
|
|
#include "char/char.h"
|
2004-07-14 17:28:59 +00:00
|
|
|
|
2006-03-11 15:35:30 +00:00
|
|
|
//#define DEBUG_CONSOLE
|
2004-07-14 17:28:59 +00:00
|
|
|
#define DEFAULT_BACKSCROLL 512
|
|
|
|
#define MAX_CONSOLES 12
|
2012-07-10 22:00:55 +02:00
|
|
|
#define CONSOLE_CURSOR_PERIOD 500
|
2004-07-14 17:28:59 +00:00
|
|
|
|
2006-06-25 17:37:36 +00:00
|
|
|
#define QEMU_RGBA(r, g, b, a) (((a) << 24) | ((r) << 16) | ((g) << 8) | (b))
|
|
|
|
#define QEMU_RGB(r, g, b) QEMU_RGBA(r, g, b, 0xff)
|
2004-07-14 17:28:59 +00:00
|
|
|
|
2006-03-11 15:35:30 +00:00
|
|
|
typedef struct TextAttributes {
|
|
|
|
uint8_t fgcol:4;
|
|
|
|
uint8_t bgcol:4;
|
|
|
|
uint8_t bold:1;
|
|
|
|
uint8_t uline:1;
|
|
|
|
uint8_t blink:1;
|
|
|
|
uint8_t invers:1;
|
|
|
|
uint8_t unvisible:1;
|
|
|
|
} TextAttributes;
|
|
|
|
|
2004-07-14 17:28:59 +00:00
|
|
|
typedef struct TextCell {
|
|
|
|
uint8_t ch;
|
2006-03-11 15:35:30 +00:00
|
|
|
TextAttributes t_attrib;
|
2004-07-14 17:28:59 +00:00
|
|
|
} TextCell;
|
|
|
|
|
|
|
|
#define MAX_ESC_PARAMS 3
|
|
|
|
|
|
|
|
enum TTYState {
|
|
|
|
TTY_STATE_NORM,
|
|
|
|
TTY_STATE_ESC,
|
|
|
|
TTY_STATE_CSI,
|
|
|
|
};
|
|
|
|
|
2006-06-25 16:26:29 +00:00
|
|
|
typedef struct QEMUFIFO {
|
|
|
|
uint8_t *buf;
|
|
|
|
int buf_size;
|
|
|
|
int count, wptr, rptr;
|
|
|
|
} QEMUFIFO;
|
|
|
|
|
2007-11-18 01:44:38 +00:00
|
|
|
static int qemu_fifo_write(QEMUFIFO *f, const uint8_t *buf, int len1)
|
2006-06-25 16:26:29 +00:00
|
|
|
{
|
|
|
|
int l, len;
|
|
|
|
|
|
|
|
l = f->buf_size - f->count;
|
|
|
|
if (len1 > l)
|
|
|
|
len1 = l;
|
|
|
|
len = len1;
|
|
|
|
while (len > 0) {
|
|
|
|
l = f->buf_size - f->wptr;
|
|
|
|
if (l > len)
|
|
|
|
l = len;
|
|
|
|
memcpy(f->buf + f->wptr, buf, l);
|
|
|
|
f->wptr += l;
|
|
|
|
if (f->wptr >= f->buf_size)
|
|
|
|
f->wptr = 0;
|
|
|
|
buf += l;
|
|
|
|
len -= l;
|
|
|
|
}
|
|
|
|
f->count += len1;
|
|
|
|
return len1;
|
|
|
|
}
|
|
|
|
|
2007-11-18 01:44:38 +00:00
|
|
|
static int qemu_fifo_read(QEMUFIFO *f, uint8_t *buf, int len1)
|
2006-06-25 16:26:29 +00:00
|
|
|
{
|
|
|
|
int l, len;
|
|
|
|
|
|
|
|
if (len1 > f->count)
|
|
|
|
len1 = f->count;
|
|
|
|
len = len1;
|
|
|
|
while (len > 0) {
|
|
|
|
l = f->buf_size - f->rptr;
|
|
|
|
if (l > len)
|
|
|
|
l = len;
|
|
|
|
memcpy(buf, f->buf + f->rptr, l);
|
|
|
|
f->rptr += l;
|
|
|
|
if (f->rptr >= f->buf_size)
|
|
|
|
f->rptr = 0;
|
|
|
|
buf += l;
|
|
|
|
len -= l;
|
|
|
|
}
|
|
|
|
f->count -= len1;
|
|
|
|
return len1;
|
|
|
|
}
|
|
|
|
|
2007-07-11 23:14:59 +00:00
|
|
|
typedef enum {
|
|
|
|
GRAPHIC_CONSOLE,
|
2008-09-24 03:32:33 +00:00
|
|
|
TEXT_CONSOLE,
|
|
|
|
TEXT_CONSOLE_FIXED_SIZE
|
2009-10-01 16:12:16 -05:00
|
|
|
} console_type_t;
|
2007-07-11 23:14:59 +00:00
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
struct QemuConsole {
|
2011-09-16 00:48:07 +02:00
|
|
|
int index;
|
2009-10-01 16:12:16 -05:00
|
|
|
console_type_t console_type;
|
2004-07-14 17:28:59 +00:00
|
|
|
DisplayState *ds;
|
2012-09-28 13:24:17 +02:00
|
|
|
|
2006-04-09 01:06:34 +00:00
|
|
|
/* Graphic console state. */
|
|
|
|
vga_hw_update_ptr hw_update;
|
|
|
|
vga_hw_invalidate_ptr hw_invalidate;
|
|
|
|
vga_hw_screen_dump_ptr hw_screen_dump;
|
2008-02-10 16:33:14 +00:00
|
|
|
vga_hw_text_update_ptr hw_text_update;
|
2006-04-09 01:06:34 +00:00
|
|
|
void *hw;
|
2004-07-14 17:28:59 +00:00
|
|
|
int g_width, g_height;
|
2012-09-28 13:24:17 +02:00
|
|
|
|
|
|
|
/* Text console state */
|
2004-07-14 17:28:59 +00:00
|
|
|
int width;
|
|
|
|
int height;
|
|
|
|
int total_height;
|
|
|
|
int backscroll_height;
|
|
|
|
int x, y;
|
2007-01-16 23:02:36 +00:00
|
|
|
int x_saved, y_saved;
|
2004-07-14 17:28:59 +00:00
|
|
|
int y_displayed;
|
|
|
|
int y_base;
|
2006-03-11 15:35:30 +00:00
|
|
|
TextAttributes t_attrib_default; /* default text attributes */
|
|
|
|
TextAttributes t_attrib; /* currently active text attributes */
|
2004-07-14 17:28:59 +00:00
|
|
|
TextCell *cells;
|
2008-02-10 16:33:14 +00:00
|
|
|
int text_x[2], text_y[2], cursor_invalidate;
|
2010-12-23 13:42:52 +01:00
|
|
|
int echo;
|
2012-07-10 22:00:55 +02:00
|
|
|
bool cursor_visible_phase;
|
|
|
|
QEMUTimer *cursor_timer;
|
2004-07-14 17:28:59 +00:00
|
|
|
|
2009-01-21 03:02:52 +00:00
|
|
|
int update_x0;
|
|
|
|
int update_y0;
|
|
|
|
int update_x1;
|
|
|
|
int update_y1;
|
|
|
|
|
2004-07-14 17:28:59 +00:00
|
|
|
enum TTYState state;
|
|
|
|
int esc_params[MAX_ESC_PARAMS];
|
|
|
|
int nb_esc_params;
|
|
|
|
|
2007-01-27 23:46:43 +00:00
|
|
|
CharDriverState *chr;
|
2006-06-25 16:26:29 +00:00
|
|
|
/* fifo for key pressed */
|
|
|
|
QEMUFIFO out_fifo;
|
|
|
|
uint8_t out_fifo_buf[16];
|
|
|
|
QEMUTimer *kbd_timer;
|
2004-07-14 17:28:59 +00:00
|
|
|
};
|
|
|
|
|
2010-02-11 00:29:57 +01:00
|
|
|
static DisplayState *display_state;
|
2012-09-28 13:24:17 +02:00
|
|
|
static QemuConsole *active_console;
|
|
|
|
static QemuConsole *consoles[MAX_CONSOLES];
|
2004-07-14 17:28:59 +00:00
|
|
|
static int nb_consoles = 0;
|
|
|
|
|
2006-04-09 01:06:34 +00:00
|
|
|
void vga_hw_update(void)
|
|
|
|
{
|
2007-01-16 23:02:36 +00:00
|
|
|
if (active_console && active_console->hw_update)
|
2006-04-09 01:06:34 +00:00
|
|
|
active_console->hw_update(active_console->hw);
|
|
|
|
}
|
|
|
|
|
|
|
|
void vga_hw_invalidate(void)
|
|
|
|
{
|
2010-05-20 15:23:06 +02:00
|
|
|
if (active_console && active_console->hw_invalidate)
|
2006-04-09 01:06:34 +00:00
|
|
|
active_console->hw_invalidate(active_console->hw);
|
|
|
|
}
|
|
|
|
|
2012-05-24 13:48:23 -03:00
|
|
|
void qmp_screendump(const char *filename, Error **errp)
|
2006-04-09 01:06:34 +00:00
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *previous_active_console;
|
2012-02-24 12:43:45 +01:00
|
|
|
bool cswitch;
|
2008-07-19 13:04:26 +00:00
|
|
|
|
|
|
|
previous_active_console = active_console;
|
2012-02-24 12:43:45 +01:00
|
|
|
cswitch = previous_active_console && previous_active_console->index != 0;
|
2011-09-16 00:48:07 +02:00
|
|
|
|
2008-07-19 13:04:26 +00:00
|
|
|
/* There is currently no way of specifying which screen we want to dump,
|
2008-09-02 00:09:16 +00:00
|
|
|
so always dump the first one. */
|
2012-02-24 12:43:45 +01:00
|
|
|
if (cswitch) {
|
|
|
|
console_select(0);
|
|
|
|
}
|
2011-09-16 00:48:07 +02:00
|
|
|
if (consoles[0] && consoles[0]->hw_screen_dump) {
|
2012-05-24 13:48:23 -03:00
|
|
|
consoles[0]->hw_screen_dump(consoles[0]->hw, filename, cswitch, errp);
|
2012-02-24 12:43:44 +01:00
|
|
|
} else {
|
error: Strip trailing '\n' from error string arguments (again)
Commit 6daf194d and be62a2eb got rid of a bunch, but they keep coming
back. Tracked down with this Coccinelle semantic patch:
@r@
expression err, eno, cls, fmt;
position p;
@@
(
error_report(fmt, ...)@p
|
error_set(err, cls, fmt, ...)@p
|
error_set_errno(err, eno, cls, fmt, ...)@p
|
error_setg(err, fmt, ...)@p
|
error_setg_errno(err, eno, fmt, ...)@p
)
@script:python@
fmt << r.fmt;
p << r.p;
@@
if "\\n" in str(fmt):
print "%s:%s:%s:%s" % (p[0].file, p[0].line, p[0].column, fmt)
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Message-id: 1360354939-10994-4-git-send-email-armbru@redhat.com
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2013-02-08 21:22:16 +01:00
|
|
|
error_setg(errp, "device doesn't support screendump");
|
2011-09-16 00:48:07 +02:00
|
|
|
}
|
|
|
|
|
2012-02-24 12:43:45 +01:00
|
|
|
if (cswitch) {
|
2011-11-18 16:41:59 +01:00
|
|
|
console_select(previous_active_console->index);
|
|
|
|
}
|
2006-04-09 01:06:34 +00:00
|
|
|
}
|
|
|
|
|
2009-10-01 16:12:16 -05:00
|
|
|
void vga_hw_text_update(console_ch_t *chardata)
|
2008-02-10 16:33:14 +00:00
|
|
|
{
|
|
|
|
if (active_console && active_console->hw_text_update)
|
|
|
|
active_console->hw_text_update(active_console->hw, chardata);
|
|
|
|
}
|
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
static void vga_fill_rect(QemuConsole *con,
|
|
|
|
int posx, int posy, int width, int height,
|
|
|
|
uint32_t color)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
2013-03-06 13:40:47 +01:00
|
|
|
DisplaySurface *surface = qemu_console_surface(con);
|
2004-07-14 17:28:59 +00:00
|
|
|
uint8_t *d, *d1;
|
|
|
|
int x, y, bpp;
|
2007-09-17 08:09:54 +00:00
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
bpp = surface_bytes_per_pixel(surface);
|
|
|
|
d1 = surface_data(surface) +
|
|
|
|
surface_stride(surface) * posy + bpp * posx;
|
2004-07-14 17:28:59 +00:00
|
|
|
for (y = 0; y < height; y++) {
|
|
|
|
d = d1;
|
|
|
|
switch(bpp) {
|
|
|
|
case 1:
|
|
|
|
for (x = 0; x < width; x++) {
|
|
|
|
*((uint8_t *)d) = color;
|
|
|
|
d++;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
for (x = 0; x < width; x++) {
|
|
|
|
*((uint16_t *)d) = color;
|
|
|
|
d += 2;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
for (x = 0; x < width; x++) {
|
|
|
|
*((uint32_t *)d) = color;
|
|
|
|
d += 4;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2013-03-06 13:40:47 +01:00
|
|
|
d1 += surface_stride(surface);
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* copy from (xs, ys) to (xd, yd) a rectangle of size (w, h) */
|
2013-03-06 13:40:47 +01:00
|
|
|
static void vga_bitblt(QemuConsole *con,
|
|
|
|
int xs, int ys, int xd, int yd, int w, int h)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
2013-03-06 13:40:47 +01:00
|
|
|
DisplaySurface *surface = qemu_console_surface(con);
|
2004-07-14 17:28:59 +00:00
|
|
|
const uint8_t *s;
|
|
|
|
uint8_t *d;
|
|
|
|
int wb, y, bpp;
|
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
bpp = surface_bytes_per_pixel(surface);
|
2004-07-14 17:28:59 +00:00
|
|
|
wb = w * bpp;
|
|
|
|
if (yd <= ys) {
|
2013-03-06 13:40:47 +01:00
|
|
|
s = surface_data(surface) +
|
|
|
|
surface_stride(surface) * ys + bpp * xs;
|
|
|
|
d = surface_data(surface) +
|
|
|
|
surface_stride(surface) * yd + bpp * xd;
|
2004-07-14 17:28:59 +00:00
|
|
|
for (y = 0; y < h; y++) {
|
|
|
|
memmove(d, s, wb);
|
2013-03-06 13:40:47 +01:00
|
|
|
d += surface_stride(surface);
|
|
|
|
s += surface_stride(surface);
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
} else {
|
2013-03-06 13:40:47 +01:00
|
|
|
s = surface_data(surface) +
|
|
|
|
surface_stride(surface) * (ys + h - 1) + bpp * xs;
|
|
|
|
d = surface_data(surface) +
|
|
|
|
surface_stride(surface) * (yd + h - 1) + bpp * xd;
|
2004-07-14 17:28:59 +00:00
|
|
|
for (y = 0; y < h; y++) {
|
|
|
|
memmove(d, s, wb);
|
2013-03-06 13:40:47 +01:00
|
|
|
d -= surface_stride(surface);
|
|
|
|
s -= surface_stride(surface);
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************/
|
|
|
|
/* basic char display */
|
|
|
|
|
|
|
|
#define FONT_HEIGHT 16
|
|
|
|
#define FONT_WIDTH 8
|
|
|
|
|
|
|
|
#include "vgafont.h"
|
|
|
|
|
|
|
|
#define cbswap_32(__x) \
|
|
|
|
((uint32_t)( \
|
|
|
|
(((uint32_t)(__x) & (uint32_t)0x000000ffUL) << 24) | \
|
|
|
|
(((uint32_t)(__x) & (uint32_t)0x0000ff00UL) << 8) | \
|
|
|
|
(((uint32_t)(__x) & (uint32_t)0x00ff0000UL) >> 8) | \
|
|
|
|
(((uint32_t)(__x) & (uint32_t)0xff000000UL) >> 24) ))
|
|
|
|
|
2009-07-27 16:13:06 +02:00
|
|
|
#ifdef HOST_WORDS_BIGENDIAN
|
2004-07-14 17:28:59 +00:00
|
|
|
#define PAT(x) x
|
|
|
|
#else
|
|
|
|
#define PAT(x) cbswap_32(x)
|
|
|
|
#endif
|
|
|
|
|
|
|
|
static const uint32_t dmask16[16] = {
|
|
|
|
PAT(0x00000000),
|
|
|
|
PAT(0x000000ff),
|
|
|
|
PAT(0x0000ff00),
|
|
|
|
PAT(0x0000ffff),
|
|
|
|
PAT(0x00ff0000),
|
|
|
|
PAT(0x00ff00ff),
|
|
|
|
PAT(0x00ffff00),
|
|
|
|
PAT(0x00ffffff),
|
|
|
|
PAT(0xff000000),
|
|
|
|
PAT(0xff0000ff),
|
|
|
|
PAT(0xff00ff00),
|
|
|
|
PAT(0xff00ffff),
|
|
|
|
PAT(0xffff0000),
|
|
|
|
PAT(0xffff00ff),
|
|
|
|
PAT(0xffffff00),
|
|
|
|
PAT(0xffffffff),
|
|
|
|
};
|
|
|
|
|
|
|
|
static const uint32_t dmask4[4] = {
|
|
|
|
PAT(0x00000000),
|
|
|
|
PAT(0x0000ffff),
|
|
|
|
PAT(0xffff0000),
|
|
|
|
PAT(0xffffffff),
|
|
|
|
};
|
|
|
|
|
2011-09-07 15:44:36 -04:00
|
|
|
#ifndef CONFIG_CURSES
|
2006-03-11 15:35:30 +00:00
|
|
|
enum color_names {
|
|
|
|
COLOR_BLACK = 0,
|
|
|
|
COLOR_RED = 1,
|
|
|
|
COLOR_GREEN = 2,
|
|
|
|
COLOR_YELLOW = 3,
|
|
|
|
COLOR_BLUE = 4,
|
|
|
|
COLOR_MAGENTA = 5,
|
|
|
|
COLOR_CYAN = 6,
|
|
|
|
COLOR_WHITE = 7
|
|
|
|
};
|
2011-09-07 15:44:36 -04:00
|
|
|
#endif
|
2006-03-11 15:35:30 +00:00
|
|
|
|
|
|
|
static const uint32_t color_table_rgb[2][8] = {
|
|
|
|
{ /* dark */
|
2006-06-25 17:37:36 +00:00
|
|
|
QEMU_RGB(0x00, 0x00, 0x00), /* black */
|
|
|
|
QEMU_RGB(0xaa, 0x00, 0x00), /* red */
|
|
|
|
QEMU_RGB(0x00, 0xaa, 0x00), /* green */
|
|
|
|
QEMU_RGB(0xaa, 0xaa, 0x00), /* yellow */
|
|
|
|
QEMU_RGB(0x00, 0x00, 0xaa), /* blue */
|
|
|
|
QEMU_RGB(0xaa, 0x00, 0xaa), /* magenta */
|
|
|
|
QEMU_RGB(0x00, 0xaa, 0xaa), /* cyan */
|
|
|
|
QEMU_RGB(0xaa, 0xaa, 0xaa), /* white */
|
2006-03-11 15:35:30 +00:00
|
|
|
},
|
|
|
|
{ /* bright */
|
2006-06-25 17:37:36 +00:00
|
|
|
QEMU_RGB(0x00, 0x00, 0x00), /* black */
|
|
|
|
QEMU_RGB(0xff, 0x00, 0x00), /* red */
|
|
|
|
QEMU_RGB(0x00, 0xff, 0x00), /* green */
|
|
|
|
QEMU_RGB(0xff, 0xff, 0x00), /* yellow */
|
|
|
|
QEMU_RGB(0x00, 0x00, 0xff), /* blue */
|
|
|
|
QEMU_RGB(0xff, 0x00, 0xff), /* magenta */
|
|
|
|
QEMU_RGB(0x00, 0xff, 0xff), /* cyan */
|
|
|
|
QEMU_RGB(0xff, 0xff, 0xff), /* white */
|
2006-03-11 15:35:30 +00:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
};
|
|
|
|
|
2006-03-11 15:35:30 +00:00
|
|
|
#ifdef DEBUG_CONSOLE
|
|
|
|
static void console_print_text_attributes(TextAttributes *t_attrib, char ch)
|
|
|
|
{
|
|
|
|
if (t_attrib->bold) {
|
|
|
|
printf("b");
|
|
|
|
} else {
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
if (t_attrib->uline) {
|
|
|
|
printf("u");
|
|
|
|
} else {
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
if (t_attrib->blink) {
|
|
|
|
printf("l");
|
|
|
|
} else {
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
if (t_attrib->invers) {
|
|
|
|
printf("i");
|
|
|
|
} else {
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
if (t_attrib->unvisible) {
|
|
|
|
printf("n");
|
|
|
|
} else {
|
|
|
|
printf(" ");
|
|
|
|
}
|
|
|
|
|
|
|
|
printf(" fg: %d bg: %d ch:'%2X' '%c'\n", t_attrib->fgcol, t_attrib->bgcol, ch, ch);
|
|
|
|
}
|
|
|
|
#endif
|
2004-07-14 17:28:59 +00:00
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
static void vga_putcharxy(QemuConsole *s, int x, int y, int ch,
|
2006-03-11 15:35:30 +00:00
|
|
|
TextAttributes *t_attrib)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
2013-03-06 13:40:47 +01:00
|
|
|
DisplaySurface *surface = qemu_console_surface(s);
|
2004-07-14 17:28:59 +00:00
|
|
|
uint8_t *d;
|
|
|
|
const uint8_t *font_ptr;
|
|
|
|
unsigned int font_data, linesize, xorcol, bpp;
|
|
|
|
int i;
|
2006-03-11 15:35:30 +00:00
|
|
|
unsigned int fgcol, bgcol;
|
|
|
|
|
|
|
|
#ifdef DEBUG_CONSOLE
|
|
|
|
printf("x: %2i y: %2i", x, y);
|
|
|
|
console_print_text_attributes(t_attrib, ch);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
if (t_attrib->invers) {
|
2013-03-06 09:50:51 +01:00
|
|
|
bgcol = color_table_rgb[t_attrib->bold][t_attrib->fgcol];
|
|
|
|
fgcol = color_table_rgb[t_attrib->bold][t_attrib->bgcol];
|
2006-03-11 15:35:30 +00:00
|
|
|
} else {
|
2013-03-06 09:50:51 +01:00
|
|
|
fgcol = color_table_rgb[t_attrib->bold][t_attrib->fgcol];
|
|
|
|
bgcol = color_table_rgb[t_attrib->bold][t_attrib->bgcol];
|
2006-03-11 15:35:30 +00:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
bpp = surface_bytes_per_pixel(surface);
|
|
|
|
d = surface_data(surface) +
|
|
|
|
surface_stride(surface) * y * FONT_HEIGHT + bpp * x * FONT_WIDTH;
|
|
|
|
linesize = surface_stride(surface);
|
2004-07-14 17:28:59 +00:00
|
|
|
font_ptr = vgafont16 + FONT_HEIGHT * ch;
|
|
|
|
xorcol = bgcol ^ fgcol;
|
2013-03-06 13:40:47 +01:00
|
|
|
switch (surface_bits_per_pixel(surface)) {
|
2004-07-14 17:28:59 +00:00
|
|
|
case 8:
|
|
|
|
for(i = 0; i < FONT_HEIGHT; i++) {
|
|
|
|
font_data = *font_ptr++;
|
2006-03-11 15:35:30 +00:00
|
|
|
if (t_attrib->uline
|
|
|
|
&& ((i == FONT_HEIGHT - 2) || (i == FONT_HEIGHT - 3))) {
|
2011-11-04 10:38:29 +01:00
|
|
|
font_data = 0xFF;
|
2006-03-11 15:35:30 +00:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
((uint32_t *)d)[0] = (dmask16[(font_data >> 4)] & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[1] = (dmask16[(font_data >> 0) & 0xf] & xorcol) ^ bgcol;
|
|
|
|
d += linesize;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 16:
|
|
|
|
case 15:
|
|
|
|
for(i = 0; i < FONT_HEIGHT; i++) {
|
|
|
|
font_data = *font_ptr++;
|
2006-03-11 15:35:30 +00:00
|
|
|
if (t_attrib->uline
|
|
|
|
&& ((i == FONT_HEIGHT - 2) || (i == FONT_HEIGHT - 3))) {
|
2011-11-04 10:38:29 +01:00
|
|
|
font_data = 0xFF;
|
2006-03-11 15:35:30 +00:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
((uint32_t *)d)[0] = (dmask4[(font_data >> 6)] & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[1] = (dmask4[(font_data >> 4) & 3] & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[2] = (dmask4[(font_data >> 2) & 3] & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[3] = (dmask4[(font_data >> 0) & 3] & xorcol) ^ bgcol;
|
|
|
|
d += linesize;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 32:
|
|
|
|
for(i = 0; i < FONT_HEIGHT; i++) {
|
|
|
|
font_data = *font_ptr++;
|
2006-03-11 15:35:30 +00:00
|
|
|
if (t_attrib->uline && ((i == FONT_HEIGHT - 2) || (i == FONT_HEIGHT - 3))) {
|
2011-11-04 10:38:29 +01:00
|
|
|
font_data = 0xFF;
|
2006-03-11 15:35:30 +00:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
((uint32_t *)d)[0] = (-((font_data >> 7)) & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[1] = (-((font_data >> 6) & 1) & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[2] = (-((font_data >> 5) & 1) & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[3] = (-((font_data >> 4) & 1) & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[4] = (-((font_data >> 3) & 1) & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[5] = (-((font_data >> 2) & 1) & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[6] = (-((font_data >> 1) & 1) & xorcol) ^ bgcol;
|
|
|
|
((uint32_t *)d)[7] = (-((font_data >> 0) & 1) & xorcol) ^ bgcol;
|
|
|
|
d += linesize;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static void text_console_resize(QemuConsole *s)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
|
|
|
TextCell *cells, *c, *c1;
|
|
|
|
int w1, x, y, last_width;
|
|
|
|
|
|
|
|
last_width = s->width;
|
|
|
|
s->width = s->g_width / FONT_WIDTH;
|
|
|
|
s->height = s->g_height / FONT_HEIGHT;
|
|
|
|
|
|
|
|
w1 = last_width;
|
|
|
|
if (s->width < w1)
|
|
|
|
w1 = s->width;
|
|
|
|
|
2011-08-20 22:09:37 -05:00
|
|
|
cells = g_malloc(s->width * s->total_height * sizeof(TextCell));
|
2004-07-14 17:28:59 +00:00
|
|
|
for(y = 0; y < s->total_height; y++) {
|
|
|
|
c = &cells[y * s->width];
|
|
|
|
if (w1 > 0) {
|
|
|
|
c1 = &s->cells[y * last_width];
|
|
|
|
for(x = 0; x < w1; x++) {
|
|
|
|
*c++ = *c1++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(x = w1; x < s->width; x++) {
|
|
|
|
c->ch = ' ';
|
2006-03-11 15:35:30 +00:00
|
|
|
c->t_attrib = s->t_attrib_default;
|
2004-07-14 17:28:59 +00:00
|
|
|
c++;
|
|
|
|
}
|
|
|
|
}
|
2011-08-20 22:09:37 -05:00
|
|
|
g_free(s->cells);
|
2004-07-14 17:28:59 +00:00
|
|
|
s->cells = cells;
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static inline void text_update_xy(QemuConsole *s, int x, int y)
|
2008-02-10 16:33:14 +00:00
|
|
|
{
|
|
|
|
s->text_x[0] = MIN(s->text_x[0], x);
|
|
|
|
s->text_x[1] = MAX(s->text_x[1], x);
|
|
|
|
s->text_y[0] = MIN(s->text_y[0], y);
|
|
|
|
s->text_y[1] = MAX(s->text_y[1], y);
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static void invalidate_xy(QemuConsole *s, int x, int y)
|
2009-01-21 03:02:52 +00:00
|
|
|
{
|
|
|
|
if (s->update_x0 > x * FONT_WIDTH)
|
|
|
|
s->update_x0 = x * FONT_WIDTH;
|
|
|
|
if (s->update_y0 > y * FONT_HEIGHT)
|
|
|
|
s->update_y0 = y * FONT_HEIGHT;
|
|
|
|
if (s->update_x1 < (x + 1) * FONT_WIDTH)
|
|
|
|
s->update_x1 = (x + 1) * FONT_WIDTH;
|
|
|
|
if (s->update_y1 < (y + 1) * FONT_HEIGHT)
|
|
|
|
s->update_y1 = (y + 1) * FONT_HEIGHT;
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static void update_xy(QemuConsole *s, int x, int y)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
|
|
|
TextCell *c;
|
|
|
|
int y1, y2;
|
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
if (s != active_console) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (s->ds->have_text) {
|
|
|
|
text_update_xy(s, x, y);
|
|
|
|
}
|
2008-02-10 16:33:14 +00:00
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
if (s->ds->have_gfx) {
|
2004-07-14 17:28:59 +00:00
|
|
|
y1 = (s->y_base + y) % s->total_height;
|
|
|
|
y2 = y1 - s->y_displayed;
|
|
|
|
if (y2 < 0)
|
|
|
|
y2 += s->total_height;
|
|
|
|
if (y2 < s->height) {
|
|
|
|
c = &s->cells[y1 * s->width + x];
|
2013-03-06 13:40:47 +01:00
|
|
|
vga_putcharxy(s, x, y2, c->ch,
|
2006-03-11 15:35:30 +00:00
|
|
|
&(c->t_attrib));
|
2009-01-21 03:02:52 +00:00
|
|
|
invalidate_xy(s, x, y2);
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static void console_show_cursor(QemuConsole *s, int show)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
|
|
|
TextCell *c;
|
|
|
|
int y, y1;
|
2013-03-06 13:40:47 +01:00
|
|
|
int x = s->x;
|
2004-07-14 17:28:59 +00:00
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
if (s != active_console) {
|
|
|
|
return;
|
|
|
|
}
|
2008-02-10 16:33:14 +00:00
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
if (s->ds->have_text) {
|
|
|
|
s->cursor_invalidate = 1;
|
|
|
|
}
|
2008-02-10 16:33:14 +00:00
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
if (s->ds->have_gfx) {
|
2007-02-10 22:37:56 +00:00
|
|
|
if (x >= s->width) {
|
|
|
|
x = s->width - 1;
|
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
y1 = (s->y_base + s->y) % s->total_height;
|
|
|
|
y = y1 - s->y_displayed;
|
|
|
|
if (y < 0)
|
|
|
|
y += s->total_height;
|
|
|
|
if (y < s->height) {
|
2007-02-10 22:37:56 +00:00
|
|
|
c = &s->cells[y1 * s->width + x];
|
2012-07-10 22:00:55 +02:00
|
|
|
if (show && s->cursor_visible_phase) {
|
2006-03-11 15:35:30 +00:00
|
|
|
TextAttributes t_attrib = s->t_attrib_default;
|
|
|
|
t_attrib.invers = !(t_attrib.invers); /* invert fg and bg */
|
2013-03-06 13:40:47 +01:00
|
|
|
vga_putcharxy(s, x, y, c->ch, &t_attrib);
|
2004-07-14 17:28:59 +00:00
|
|
|
} else {
|
2013-03-06 13:40:47 +01:00
|
|
|
vga_putcharxy(s, x, y, c->ch, &(c->t_attrib));
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
2009-01-21 03:02:52 +00:00
|
|
|
invalidate_xy(s, x, y);
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static void console_refresh(QemuConsole *s)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
2013-03-06 13:40:47 +01:00
|
|
|
DisplaySurface *surface = qemu_console_surface(s);
|
2004-07-14 17:28:59 +00:00
|
|
|
TextCell *c;
|
|
|
|
int x, y, y1;
|
|
|
|
|
2007-09-16 21:08:06 +00:00
|
|
|
if (s != active_console)
|
2004-07-14 17:28:59 +00:00
|
|
|
return;
|
2012-09-28 15:02:08 +02:00
|
|
|
|
|
|
|
if (s->ds->have_text) {
|
2008-02-10 16:33:14 +00:00
|
|
|
s->text_x[0] = 0;
|
|
|
|
s->text_y[0] = 0;
|
|
|
|
s->text_x[1] = s->width - 1;
|
|
|
|
s->text_y[1] = s->height - 1;
|
|
|
|
s->cursor_invalidate = 1;
|
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
|
2012-09-28 15:02:08 +02:00
|
|
|
if (s->ds->have_gfx) {
|
2013-03-06 13:40:47 +01:00
|
|
|
vga_fill_rect(s, 0, 0, surface_width(surface), surface_height(surface),
|
2013-03-06 09:50:51 +01:00
|
|
|
color_table_rgb[0][COLOR_BLACK]);
|
2012-09-28 15:02:08 +02:00
|
|
|
y1 = s->y_displayed;
|
|
|
|
for (y = 0; y < s->height; y++) {
|
|
|
|
c = s->cells + y1 * s->width;
|
|
|
|
for (x = 0; x < s->width; x++) {
|
2013-03-06 13:40:47 +01:00
|
|
|
vga_putcharxy(s, x, y, c->ch,
|
2012-09-28 15:02:08 +02:00
|
|
|
&(c->t_attrib));
|
|
|
|
c++;
|
|
|
|
}
|
|
|
|
if (++y1 == s->total_height) {
|
|
|
|
y1 = 0;
|
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
2012-09-28 15:02:08 +02:00
|
|
|
console_show_cursor(s, 1);
|
2013-03-06 13:40:47 +01:00
|
|
|
dpy_gfx_update(s, 0, 0,
|
|
|
|
surface_width(surface), surface_height(surface));
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void console_scroll(int ydelta)
|
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s;
|
2004-07-14 17:28:59 +00:00
|
|
|
int i, y1;
|
2007-09-17 08:09:54 +00:00
|
|
|
|
2004-07-14 17:28:59 +00:00
|
|
|
s = active_console;
|
2007-07-11 23:14:59 +00:00
|
|
|
if (!s || (s->console_type == GRAPHIC_CONSOLE))
|
2004-07-14 17:28:59 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
if (ydelta > 0) {
|
|
|
|
for(i = 0; i < ydelta; i++) {
|
|
|
|
if (s->y_displayed == s->y_base)
|
|
|
|
break;
|
|
|
|
if (++s->y_displayed == s->total_height)
|
|
|
|
s->y_displayed = 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ydelta = -ydelta;
|
|
|
|
i = s->backscroll_height;
|
|
|
|
if (i > s->total_height - s->height)
|
|
|
|
i = s->total_height - s->height;
|
|
|
|
y1 = s->y_base - i;
|
|
|
|
if (y1 < 0)
|
|
|
|
y1 += s->total_height;
|
|
|
|
for(i = 0; i < ydelta; i++) {
|
|
|
|
if (s->y_displayed == y1)
|
|
|
|
break;
|
|
|
|
if (--s->y_displayed < 0)
|
|
|
|
s->y_displayed = s->total_height - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
console_refresh(s);
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static void console_put_lf(QemuConsole *s)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
|
|
|
TextCell *c;
|
|
|
|
int x, y1;
|
|
|
|
|
|
|
|
s->y++;
|
|
|
|
if (s->y >= s->height) {
|
|
|
|
s->y = s->height - 1;
|
2006-03-11 15:35:30 +00:00
|
|
|
|
2004-07-14 17:28:59 +00:00
|
|
|
if (s->y_displayed == s->y_base) {
|
|
|
|
if (++s->y_displayed == s->total_height)
|
|
|
|
s->y_displayed = 0;
|
|
|
|
}
|
|
|
|
if (++s->y_base == s->total_height)
|
|
|
|
s->y_base = 0;
|
|
|
|
if (s->backscroll_height < s->total_height)
|
|
|
|
s->backscroll_height++;
|
|
|
|
y1 = (s->y_base + s->height - 1) % s->total_height;
|
|
|
|
c = &s->cells[y1 * s->width];
|
|
|
|
for(x = 0; x < s->width; x++) {
|
|
|
|
c->ch = ' ';
|
2006-03-11 15:35:30 +00:00
|
|
|
c->t_attrib = s->t_attrib_default;
|
2004-07-14 17:28:59 +00:00
|
|
|
c++;
|
|
|
|
}
|
|
|
|
if (s == active_console && s->y_displayed == s->y_base) {
|
2013-03-06 13:40:47 +01:00
|
|
|
if (s->ds->have_text) {
|
2008-02-10 16:33:14 +00:00
|
|
|
s->text_x[0] = 0;
|
|
|
|
s->text_y[0] = 0;
|
|
|
|
s->text_x[1] = s->width - 1;
|
|
|
|
s->text_y[1] = s->height - 1;
|
|
|
|
}
|
|
|
|
|
2013-03-06 13:40:47 +01:00
|
|
|
if (s->ds->have_gfx) {
|
|
|
|
vga_bitblt(s, 0, FONT_HEIGHT, 0, 0,
|
|
|
|
s->width * FONT_WIDTH,
|
|
|
|
(s->height - 1) * FONT_HEIGHT);
|
|
|
|
vga_fill_rect(s, 0, (s->height - 1) * FONT_HEIGHT,
|
|
|
|
s->width * FONT_WIDTH, FONT_HEIGHT,
|
|
|
|
color_table_rgb[0][s->t_attrib_default.bgcol]);
|
|
|
|
s->update_x0 = 0;
|
|
|
|
s->update_y0 = 0;
|
|
|
|
s->update_x1 = s->width * FONT_WIDTH;
|
|
|
|
s->update_y1 = s->height * FONT_HEIGHT;
|
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-03-11 15:35:30 +00:00
|
|
|
/* Set console attributes depending on the current escape codes.
|
|
|
|
* NOTE: I know this code is not very efficient (checking every color for it
|
|
|
|
* self) but it is more readable and better maintainable.
|
|
|
|
*/
|
2012-09-28 13:24:17 +02:00
|
|
|
static void console_handle_escape(QemuConsole *s)
|
2006-03-11 15:35:30 +00:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i=0; i<s->nb_esc_params; i++) {
|
|
|
|
switch (s->esc_params[i]) {
|
|
|
|
case 0: /* reset all console attributes to default */
|
|
|
|
s->t_attrib = s->t_attrib_default;
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
s->t_attrib.bold = 1;
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
s->t_attrib.uline = 1;
|
|
|
|
break;
|
|
|
|
case 5:
|
|
|
|
s->t_attrib.blink = 1;
|
|
|
|
break;
|
|
|
|
case 7:
|
|
|
|
s->t_attrib.invers = 1;
|
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
s->t_attrib.unvisible = 1;
|
|
|
|
break;
|
|
|
|
case 22:
|
|
|
|
s->t_attrib.bold = 0;
|
|
|
|
break;
|
|
|
|
case 24:
|
|
|
|
s->t_attrib.uline = 0;
|
|
|
|
break;
|
|
|
|
case 25:
|
|
|
|
s->t_attrib.blink = 0;
|
|
|
|
break;
|
|
|
|
case 27:
|
|
|
|
s->t_attrib.invers = 0;
|
|
|
|
break;
|
|
|
|
case 28:
|
|
|
|
s->t_attrib.unvisible = 0;
|
|
|
|
break;
|
|
|
|
/* set foreground color */
|
|
|
|
case 30:
|
|
|
|
s->t_attrib.fgcol=COLOR_BLACK;
|
|
|
|
break;
|
|
|
|
case 31:
|
|
|
|
s->t_attrib.fgcol=COLOR_RED;
|
|
|
|
break;
|
|
|
|
case 32:
|
|
|
|
s->t_attrib.fgcol=COLOR_GREEN;
|
|
|
|
break;
|
|
|
|
case 33:
|
|
|
|
s->t_attrib.fgcol=COLOR_YELLOW;
|
|
|
|
break;
|
|
|
|
case 34:
|
|
|
|
s->t_attrib.fgcol=COLOR_BLUE;
|
|
|
|
break;
|
|
|
|
case 35:
|
|
|
|
s->t_attrib.fgcol=COLOR_MAGENTA;
|
|
|
|
break;
|
|
|
|
case 36:
|
|
|
|
s->t_attrib.fgcol=COLOR_CYAN;
|
|
|
|
break;
|
|
|
|
case 37:
|
|
|
|
s->t_attrib.fgcol=COLOR_WHITE;
|
|
|
|
break;
|
|
|
|
/* set background color */
|
|
|
|
case 40:
|
|
|
|
s->t_attrib.bgcol=COLOR_BLACK;
|
|
|
|
break;
|
|
|
|
case 41:
|
|
|
|
s->t_attrib.bgcol=COLOR_RED;
|
|
|
|
break;
|
|
|
|
case 42:
|
|
|
|
s->t_attrib.bgcol=COLOR_GREEN;
|
|
|
|
break;
|
|
|
|
case 43:
|
|
|
|
s->t_attrib.bgcol=COLOR_YELLOW;
|
|
|
|
break;
|
|
|
|
case 44:
|
|
|
|
s->t_attrib.bgcol=COLOR_BLUE;
|
|
|
|
break;
|
|
|
|
case 45:
|
|
|
|
s->t_attrib.bgcol=COLOR_MAGENTA;
|
|
|
|
break;
|
|
|
|
case 46:
|
|
|
|
s->t_attrib.bgcol=COLOR_CYAN;
|
|
|
|
break;
|
|
|
|
case 47:
|
|
|
|
s->t_attrib.bgcol=COLOR_WHITE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static void console_clear_xy(QemuConsole *s, int x, int y)
|
2007-01-16 23:02:36 +00:00
|
|
|
{
|
|
|
|
int y1 = (s->y_base + y) % s->total_height;
|
|
|
|
TextCell *c = &s->cells[y1 * s->width + x];
|
|
|
|
c->ch = ' ';
|
|
|
|
c->t_attrib = s->t_attrib_default;
|
|
|
|
update_xy(s, x, y);
|
|
|
|
}
|
|
|
|
|
2012-09-04 10:26:09 -05:00
|
|
|
/* set cursor, checking bounds */
|
2012-09-28 13:24:17 +02:00
|
|
|
static void set_cursor(QemuConsole *s, int x, int y)
|
2012-09-04 10:26:09 -05:00
|
|
|
{
|
|
|
|
if (x < 0) {
|
|
|
|
x = 0;
|
|
|
|
}
|
|
|
|
if (y < 0) {
|
|
|
|
y = 0;
|
|
|
|
}
|
|
|
|
if (y >= s->height) {
|
|
|
|
y = s->height - 1;
|
|
|
|
}
|
|
|
|
if (x >= s->width) {
|
|
|
|
x = s->width - 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
s->x = x;
|
|
|
|
s->y = y;
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static void console_putchar(QemuConsole *s, int ch)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
|
|
|
TextCell *c;
|
2007-01-16 23:02:36 +00:00
|
|
|
int y1, i;
|
|
|
|
int x, y;
|
2004-07-14 17:28:59 +00:00
|
|
|
|
|
|
|
switch(s->state) {
|
|
|
|
case TTY_STATE_NORM:
|
|
|
|
switch(ch) {
|
2006-03-11 15:35:30 +00:00
|
|
|
case '\r': /* carriage return */
|
2004-07-14 17:28:59 +00:00
|
|
|
s->x = 0;
|
|
|
|
break;
|
2006-03-11 15:35:30 +00:00
|
|
|
case '\n': /* newline */
|
2004-07-14 17:28:59 +00:00
|
|
|
console_put_lf(s);
|
|
|
|
break;
|
2006-03-11 15:35:30 +00:00
|
|
|
case '\b': /* backspace */
|
2007-09-16 21:08:06 +00:00
|
|
|
if (s->x > 0)
|
2006-06-25 16:26:29 +00:00
|
|
|
s->x--;
|
2006-03-11 15:35:30 +00:00
|
|
|
break;
|
|
|
|
case '\t': /* tabspace */
|
|
|
|
if (s->x + (8 - (s->x % 8)) > s->width) {
|
2006-07-14 20:24:31 +00:00
|
|
|
s->x = 0;
|
2006-03-11 15:35:30 +00:00
|
|
|
console_put_lf(s);
|
|
|
|
} else {
|
|
|
|
s->x = s->x + (8 - (s->x % 8));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '\a': /* alert aka. bell */
|
|
|
|
/* TODO: has to be implemented */
|
|
|
|
break;
|
2007-01-16 23:02:36 +00:00
|
|
|
case 14:
|
|
|
|
/* SI (shift in), character set 0 (ignored) */
|
|
|
|
break;
|
|
|
|
case 15:
|
|
|
|
/* SO (shift out), character set 1 (ignored) */
|
|
|
|
break;
|
2006-03-11 15:35:30 +00:00
|
|
|
case 27: /* esc (introducing an escape sequence) */
|
2004-07-14 17:28:59 +00:00
|
|
|
s->state = TTY_STATE_ESC;
|
|
|
|
break;
|
|
|
|
default:
|
2007-02-10 22:37:56 +00:00
|
|
|
if (s->x >= s->width) {
|
|
|
|
/* line wrap */
|
|
|
|
s->x = 0;
|
|
|
|
console_put_lf(s);
|
2007-01-16 23:02:36 +00:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
y1 = (s->y_base + s->y) % s->total_height;
|
|
|
|
c = &s->cells[y1 * s->width + s->x];
|
|
|
|
c->ch = ch;
|
2006-03-11 15:35:30 +00:00
|
|
|
c->t_attrib = s->t_attrib;
|
2004-07-14 17:28:59 +00:00
|
|
|
update_xy(s, s->x, s->y);
|
|
|
|
s->x++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2006-03-11 15:35:30 +00:00
|
|
|
case TTY_STATE_ESC: /* check if it is a terminal escape sequence */
|
2004-07-14 17:28:59 +00:00
|
|
|
if (ch == '[') {
|
|
|
|
for(i=0;i<MAX_ESC_PARAMS;i++)
|
|
|
|
s->esc_params[i] = 0;
|
|
|
|
s->nb_esc_params = 0;
|
|
|
|
s->state = TTY_STATE_CSI;
|
|
|
|
} else {
|
|
|
|
s->state = TTY_STATE_NORM;
|
|
|
|
}
|
|
|
|
break;
|
2006-03-11 15:35:30 +00:00
|
|
|
case TTY_STATE_CSI: /* handle escape sequence parameters */
|
2004-07-14 17:28:59 +00:00
|
|
|
if (ch >= '0' && ch <= '9') {
|
|
|
|
if (s->nb_esc_params < MAX_ESC_PARAMS) {
|
2012-09-17 11:10:03 +02:00
|
|
|
int *param = &s->esc_params[s->nb_esc_params];
|
|
|
|
int digit = (ch - '0');
|
|
|
|
|
|
|
|
*param = (*param <= (INT_MAX - digit) / 10) ?
|
|
|
|
*param * 10 + digit : INT_MAX;
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
} else {
|
2012-09-04 10:26:09 -05:00
|
|
|
if (s->nb_esc_params < MAX_ESC_PARAMS)
|
|
|
|
s->nb_esc_params++;
|
2004-07-14 17:28:59 +00:00
|
|
|
if (ch == ';')
|
|
|
|
break;
|
2007-01-16 23:02:36 +00:00
|
|
|
#ifdef DEBUG_CONSOLE
|
|
|
|
fprintf(stderr, "escape sequence CSI%d;%d%c, %d parameters\n",
|
|
|
|
s->esc_params[0], s->esc_params[1], ch, s->nb_esc_params);
|
|
|
|
#endif
|
2004-07-14 17:28:59 +00:00
|
|
|
s->state = TTY_STATE_NORM;
|
|
|
|
switch(ch) {
|
2007-01-16 23:02:36 +00:00
|
|
|
case 'A':
|
|
|
|
/* move cursor up */
|
|
|
|
if (s->esc_params[0] == 0) {
|
|
|
|
s->esc_params[0] = 1;
|
|
|
|
}
|
2012-09-04 10:26:09 -05:00
|
|
|
set_cursor(s, s->x, s->y - s->esc_params[0]);
|
2007-01-16 23:02:36 +00:00
|
|
|
break;
|
|
|
|
case 'B':
|
|
|
|
/* move cursor down */
|
|
|
|
if (s->esc_params[0] == 0) {
|
|
|
|
s->esc_params[0] = 1;
|
|
|
|
}
|
2012-09-04 10:26:09 -05:00
|
|
|
set_cursor(s, s->x, s->y + s->esc_params[0]);
|
2004-07-14 17:28:59 +00:00
|
|
|
break;
|
|
|
|
case 'C':
|
2007-01-16 23:02:36 +00:00
|
|
|
/* move cursor right */
|
|
|
|
if (s->esc_params[0] == 0) {
|
|
|
|
s->esc_params[0] = 1;
|
|
|
|
}
|
2012-09-04 10:26:09 -05:00
|
|
|
set_cursor(s, s->x + s->esc_params[0], s->y);
|
2004-07-14 17:28:59 +00:00
|
|
|
break;
|
2007-01-16 23:02:36 +00:00
|
|
|
case 'D':
|
|
|
|
/* move cursor left */
|
|
|
|
if (s->esc_params[0] == 0) {
|
|
|
|
s->esc_params[0] = 1;
|
|
|
|
}
|
2012-09-04 10:26:09 -05:00
|
|
|
set_cursor(s, s->x - s->esc_params[0], s->y);
|
2007-01-16 23:02:36 +00:00
|
|
|
break;
|
|
|
|
case 'G':
|
|
|
|
/* move cursor to column */
|
2012-09-04 10:26:09 -05:00
|
|
|
set_cursor(s, s->esc_params[0] - 1, s->y);
|
2007-01-16 23:02:36 +00:00
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
case 'H':
|
|
|
|
/* move cursor to row, column */
|
2012-09-04 10:26:09 -05:00
|
|
|
set_cursor(s, s->esc_params[1] - 1, s->esc_params[0] - 1);
|
2007-01-16 23:02:36 +00:00
|
|
|
break;
|
|
|
|
case 'J':
|
|
|
|
switch (s->esc_params[0]) {
|
|
|
|
case 0:
|
|
|
|
/* clear to end of screen */
|
|
|
|
for (y = s->y; y < s->height; y++) {
|
|
|
|
for (x = 0; x < s->width; x++) {
|
|
|
|
if (y == s->y && x < s->x) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
console_clear_xy(s, x, y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
/* clear from beginning of screen */
|
|
|
|
for (y = 0; y <= s->y; y++) {
|
|
|
|
for (x = 0; x < s->width; x++) {
|
|
|
|
if (y == s->y && x > s->x) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
console_clear_xy(s, x, y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* clear entire screen */
|
|
|
|
for (y = 0; y <= s->height; y++) {
|
|
|
|
for (x = 0; x < s->width; x++) {
|
|
|
|
console_clear_xy(s, x, y);
|
|
|
|
}
|
|
|
|
}
|
2011-11-22 11:59:06 +01:00
|
|
|
break;
|
2007-01-16 23:02:36 +00:00
|
|
|
}
|
2011-11-22 11:59:07 +01:00
|
|
|
break;
|
2004-07-14 17:28:59 +00:00
|
|
|
case 'K':
|
2007-01-16 23:02:36 +00:00
|
|
|
switch (s->esc_params[0]) {
|
|
|
|
case 0:
|
2011-11-22 11:59:06 +01:00
|
|
|
/* clear to eol */
|
|
|
|
for(x = s->x; x < s->width; x++) {
|
2007-01-16 23:02:36 +00:00
|
|
|
console_clear_xy(s, x, s->y);
|
2011-11-22 11:59:06 +01:00
|
|
|
}
|
|
|
|
break;
|
2007-01-16 23:02:36 +00:00
|
|
|
case 1:
|
|
|
|
/* clear from beginning of line */
|
|
|
|
for (x = 0; x <= s->x; x++) {
|
|
|
|
console_clear_xy(s, x, s->y);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* clear entire line */
|
|
|
|
for(x = 0; x < s->width; x++) {
|
|
|
|
console_clear_xy(s, x, s->y);
|
|
|
|
}
|
2011-11-22 11:59:06 +01:00
|
|
|
break;
|
|
|
|
}
|
2007-01-16 23:02:36 +00:00
|
|
|
break;
|
|
|
|
case 'm':
|
2011-11-22 11:59:06 +01:00
|
|
|
console_handle_escape(s);
|
|
|
|
break;
|
2007-01-16 23:02:36 +00:00
|
|
|
case 'n':
|
|
|
|
/* report cursor position */
|
|
|
|
/* TODO: send ESC[row;colR */
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
/* save cursor position */
|
|
|
|
s->x_saved = s->x;
|
|
|
|
s->y_saved = s->y;
|
|
|
|
break;
|
|
|
|
case 'u':
|
|
|
|
/* restore cursor position */
|
|
|
|
s->x = s->x_saved;
|
|
|
|
s->y = s->y_saved;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
#ifdef DEBUG_CONSOLE
|
|
|
|
fprintf(stderr, "unhandled escape character '%c'\n", ch);
|
|
|
|
#endif
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void console_select(unsigned int index)
|
|
|
|
{
|
2013-03-06 13:40:47 +01:00
|
|
|
DisplaySurface *surface;
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s;
|
2006-03-11 15:35:30 +00:00
|
|
|
|
2004-07-14 17:28:59 +00:00
|
|
|
if (index >= MAX_CONSOLES)
|
|
|
|
return;
|
2010-09-20 14:11:19 +01:00
|
|
|
if (active_console) {
|
2013-03-06 13:40:47 +01:00
|
|
|
surface = qemu_console_surface(active_console);
|
|
|
|
active_console->g_width = surface_width(surface);
|
|
|
|
active_console->g_height = surface_height(surface);
|
2010-09-20 14:11:19 +01:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
s = consoles[index];
|
|
|
|
if (s) {
|
2009-01-15 22:14:11 +00:00
|
|
|
DisplayState *ds = s->ds;
|
2012-07-10 22:00:55 +02:00
|
|
|
|
2012-08-17 15:50:44 +02:00
|
|
|
if (active_console && active_console->cursor_timer) {
|
2012-07-10 22:00:55 +02:00
|
|
|
qemu_del_timer(active_console->cursor_timer);
|
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
active_console = s;
|
2012-09-28 15:02:08 +02:00
|
|
|
if (ds->have_gfx) {
|
2013-02-28 10:48:02 +01:00
|
|
|
surface = qemu_create_displaysurface(s->g_width, s->g_height);
|
2013-03-05 15:24:14 +01:00
|
|
|
dpy_gfx_replace_surface(s, surface);
|
2012-09-28 15:02:08 +02:00
|
|
|
}
|
|
|
|
if (ds->have_text) {
|
2013-03-05 15:24:14 +01:00
|
|
|
dpy_text_resize(s, s->width, s->height);
|
2009-01-21 18:59:12 +00:00
|
|
|
}
|
2012-07-10 22:00:55 +02:00
|
|
|
if (s->cursor_timer) {
|
|
|
|
qemu_mod_timer(s->cursor_timer,
|
|
|
|
qemu_get_clock_ms(rt_clock) + CONSOLE_CURSOR_PERIOD / 2);
|
|
|
|
}
|
2008-02-10 16:33:14 +00:00
|
|
|
vga_hw_invalidate();
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int console_puts(CharDriverState *chr, const uint8_t *buf, int len)
|
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s = chr->opaque;
|
2004-07-14 17:28:59 +00:00
|
|
|
int i;
|
|
|
|
|
2009-01-21 03:02:52 +00:00
|
|
|
s->update_x0 = s->width * FONT_WIDTH;
|
|
|
|
s->update_y0 = s->height * FONT_HEIGHT;
|
|
|
|
s->update_x1 = 0;
|
|
|
|
s->update_y1 = 0;
|
2004-07-14 17:28:59 +00:00
|
|
|
console_show_cursor(s, 0);
|
|
|
|
for(i = 0; i < len; i++) {
|
|
|
|
console_putchar(s, buf[i]);
|
|
|
|
}
|
|
|
|
console_show_cursor(s, 1);
|
2012-09-28 15:02:08 +02:00
|
|
|
if (s->ds->have_gfx && s->update_x0 < s->update_x1) {
|
2013-03-05 15:24:14 +01:00
|
|
|
dpy_gfx_update(s, s->update_x0, s->update_y0,
|
2012-09-28 15:02:08 +02:00
|
|
|
s->update_x1 - s->update_x0,
|
|
|
|
s->update_y1 - s->update_y0);
|
2009-01-21 03:02:52 +00:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2006-06-25 16:26:29 +00:00
|
|
|
static void kbd_send_chars(void *opaque)
|
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s = opaque;
|
2006-06-25 16:26:29 +00:00
|
|
|
int len;
|
|
|
|
uint8_t buf[16];
|
2007-09-17 08:09:54 +00:00
|
|
|
|
2011-08-15 11:17:31 -05:00
|
|
|
len = qemu_chr_be_can_write(s->chr);
|
2006-06-25 16:26:29 +00:00
|
|
|
if (len > s->out_fifo.count)
|
|
|
|
len = s->out_fifo.count;
|
|
|
|
if (len > 0) {
|
|
|
|
if (len > sizeof(buf))
|
|
|
|
len = sizeof(buf);
|
|
|
|
qemu_fifo_read(&s->out_fifo, buf, len);
|
2011-08-15 11:17:30 -05:00
|
|
|
qemu_chr_be_write(s->chr, buf, len);
|
2006-06-25 16:26:29 +00:00
|
|
|
}
|
|
|
|
/* characters are pending: we send them a bit later (XXX:
|
|
|
|
horrible, should change char device API) */
|
|
|
|
if (s->out_fifo.count > 0) {
|
2011-03-11 16:47:48 +01:00
|
|
|
qemu_mod_timer(s->kbd_timer, qemu_get_clock_ms(rt_clock) + 1);
|
2006-06-25 16:26:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-07-14 17:28:59 +00:00
|
|
|
/* called when an ascii key is pressed */
|
|
|
|
void kbd_put_keysym(int keysym)
|
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s;
|
2004-07-14 17:28:59 +00:00
|
|
|
uint8_t buf[16], *q;
|
|
|
|
int c;
|
|
|
|
|
|
|
|
s = active_console;
|
2007-07-11 23:14:59 +00:00
|
|
|
if (!s || (s->console_type == GRAPHIC_CONSOLE))
|
2004-07-14 17:28:59 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
switch(keysym) {
|
|
|
|
case QEMU_KEY_CTRL_UP:
|
|
|
|
console_scroll(-1);
|
|
|
|
break;
|
|
|
|
case QEMU_KEY_CTRL_DOWN:
|
|
|
|
console_scroll(1);
|
|
|
|
break;
|
|
|
|
case QEMU_KEY_CTRL_PAGEUP:
|
|
|
|
console_scroll(-10);
|
|
|
|
break;
|
|
|
|
case QEMU_KEY_CTRL_PAGEDOWN:
|
|
|
|
console_scroll(10);
|
|
|
|
break;
|
|
|
|
default:
|
2006-06-25 16:26:29 +00:00
|
|
|
/* convert the QEMU keysym to VT100 key string */
|
|
|
|
q = buf;
|
|
|
|
if (keysym >= 0xe100 && keysym <= 0xe11f) {
|
|
|
|
*q++ = '\033';
|
|
|
|
*q++ = '[';
|
|
|
|
c = keysym - 0xe100;
|
|
|
|
if (c >= 10)
|
|
|
|
*q++ = '0' + (c / 10);
|
|
|
|
*q++ = '0' + (c % 10);
|
|
|
|
*q++ = '~';
|
|
|
|
} else if (keysym >= 0xe120 && keysym <= 0xe17f) {
|
|
|
|
*q++ = '\033';
|
|
|
|
*q++ = '[';
|
|
|
|
*q++ = keysym & 0xff;
|
2010-12-23 13:42:52 +01:00
|
|
|
} else if (s->echo && (keysym == '\r' || keysym == '\n')) {
|
|
|
|
console_puts(s->chr, (const uint8_t *) "\r", 1);
|
|
|
|
*q++ = '\n';
|
2006-06-25 16:26:29 +00:00
|
|
|
} else {
|
2010-12-23 13:42:52 +01:00
|
|
|
*q++ = keysym;
|
|
|
|
}
|
|
|
|
if (s->echo) {
|
|
|
|
console_puts(s->chr, buf, q - buf);
|
2006-06-25 16:26:29 +00:00
|
|
|
}
|
2007-01-27 23:46:43 +00:00
|
|
|
if (s->chr->chr_read) {
|
2006-06-25 16:26:29 +00:00
|
|
|
qemu_fifo_write(&s->out_fifo, buf, q - buf);
|
|
|
|
kbd_send_chars(s);
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-02-10 16:33:14 +00:00
|
|
|
static void text_console_invalidate(void *opaque)
|
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s = (QemuConsole *) opaque;
|
2013-03-06 13:40:47 +01:00
|
|
|
DisplaySurface *surface = qemu_console_surface(s);
|
|
|
|
|
|
|
|
if (s->ds->have_text && s->console_type == TEXT_CONSOLE) {
|
|
|
|
s->g_width = surface_width(surface);
|
|
|
|
s->g_height = surface_height(surface);
|
2009-01-21 18:59:12 +00:00
|
|
|
text_console_resize(s);
|
|
|
|
}
|
2008-02-10 16:33:14 +00:00
|
|
|
console_refresh(s);
|
|
|
|
}
|
|
|
|
|
2009-10-01 16:12:16 -05:00
|
|
|
static void text_console_update(void *opaque, console_ch_t *chardata)
|
2008-02-10 16:33:14 +00:00
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s = (QemuConsole *) opaque;
|
2008-02-10 16:33:14 +00:00
|
|
|
int i, j, src;
|
|
|
|
|
|
|
|
if (s->text_x[0] <= s->text_x[1]) {
|
|
|
|
src = (s->y_base + s->text_y[0]) * s->width;
|
|
|
|
chardata += s->text_y[0] * s->width;
|
|
|
|
for (i = s->text_y[0]; i <= s->text_y[1]; i ++)
|
|
|
|
for (j = 0; j < s->width; j ++, src ++)
|
|
|
|
console_write_ch(chardata ++, s->cells[src].ch |
|
|
|
|
(s->cells[src].t_attrib.fgcol << 12) |
|
|
|
|
(s->cells[src].t_attrib.bgcol << 8) |
|
|
|
|
(s->cells[src].t_attrib.bold << 21));
|
2013-03-05 15:24:14 +01:00
|
|
|
dpy_text_update(s, s->text_x[0], s->text_y[0],
|
2012-09-28 15:02:08 +02:00
|
|
|
s->text_x[1] - s->text_x[0], i - s->text_y[0]);
|
2008-02-10 16:33:14 +00:00
|
|
|
s->text_x[0] = s->width;
|
|
|
|
s->text_y[0] = s->height;
|
|
|
|
s->text_x[1] = 0;
|
|
|
|
s->text_y[1] = 0;
|
|
|
|
}
|
|
|
|
if (s->cursor_invalidate) {
|
2013-03-05 15:24:14 +01:00
|
|
|
dpy_text_cursor(s, s->x, s->y);
|
2008-02-10 16:33:14 +00:00
|
|
|
s->cursor_invalidate = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-28 13:24:17 +02:00
|
|
|
static QemuConsole *new_console(DisplayState *ds, console_type_t console_type)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s;
|
2006-04-09 01:06:34 +00:00
|
|
|
int i;
|
2004-07-14 17:28:59 +00:00
|
|
|
|
|
|
|
if (nb_consoles >= MAX_CONSOLES)
|
|
|
|
return NULL;
|
2012-09-28 13:24:17 +02:00
|
|
|
s = g_malloc0(sizeof(QemuConsole));
|
2007-07-11 23:14:59 +00:00
|
|
|
if (!active_console || ((active_console->console_type != GRAPHIC_CONSOLE) &&
|
|
|
|
(console_type == GRAPHIC_CONSOLE))) {
|
2004-07-14 17:28:59 +00:00
|
|
|
active_console = s;
|
2007-07-11 23:14:59 +00:00
|
|
|
}
|
2004-07-14 17:28:59 +00:00
|
|
|
s->ds = ds;
|
2007-07-11 23:14:59 +00:00
|
|
|
s->console_type = console_type;
|
|
|
|
if (console_type != GRAPHIC_CONSOLE) {
|
2011-09-16 00:48:07 +02:00
|
|
|
s->index = nb_consoles;
|
2006-04-09 01:06:34 +00:00
|
|
|
consoles[nb_consoles++] = s;
|
|
|
|
} else {
|
|
|
|
/* HACK: Put graphical consoles before text consoles. */
|
|
|
|
for (i = nb_consoles; i > 0; i--) {
|
2007-07-11 23:14:59 +00:00
|
|
|
if (consoles[i - 1]->console_type == GRAPHIC_CONSOLE)
|
2006-04-09 01:06:34 +00:00
|
|
|
break;
|
|
|
|
consoles[i] = consoles[i - 1];
|
2011-09-16 00:48:07 +02:00
|
|
|
consoles[i]->index = i;
|
2006-04-09 01:06:34 +00:00
|
|
|
}
|
2011-09-16 00:48:07 +02:00
|
|
|
s->index = i;
|
2006-04-09 01:06:34 +00:00
|
|
|
consoles[i] = s;
|
2009-01-16 19:04:14 +00:00
|
|
|
nb_consoles++;
|
2006-04-09 01:06:34 +00:00
|
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
2012-09-27 11:06:36 +02:00
|
|
|
static void qemu_alloc_display(DisplaySurface *surface, int width, int height,
|
|
|
|
int linesize, PixelFormat pf, int newflags)
|
2011-03-16 13:33:30 +01:00
|
|
|
{
|
|
|
|
surface->pf = pf;
|
2012-09-26 15:20:05 +02:00
|
|
|
|
|
|
|
qemu_pixman_image_unref(surface->image);
|
|
|
|
surface->image = NULL;
|
|
|
|
|
|
|
|
surface->format = qemu_pixman_get_format(&pf);
|
|
|
|
assert(surface->format != 0);
|
|
|
|
surface->image = pixman_image_create_bits(surface->format,
|
|
|
|
width, height,
|
|
|
|
NULL, linesize);
|
|
|
|
assert(surface->image != NULL);
|
|
|
|
|
2011-03-16 13:33:30 +01:00
|
|
|
surface->flags = newflags | QEMU_ALLOCATED_FLAG;
|
2010-02-11 00:29:57 +01:00
|
|
|
#ifdef HOST_WORDS_BIGENDIAN
|
2011-03-16 13:33:30 +01:00
|
|
|
surface->flags |= QEMU_BIG_ENDIAN_FLAG;
|
2010-02-11 00:29:57 +01:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2013-02-28 10:48:02 +01:00
|
|
|
DisplaySurface *qemu_create_displaysurface(int width, int height)
|
2012-09-27 11:06:36 +02:00
|
|
|
{
|
|
|
|
DisplaySurface *surface = g_new0(DisplaySurface, 1);
|
|
|
|
int linesize = width * 4;
|
2013-02-28 10:48:02 +01:00
|
|
|
|
|
|
|
trace_displaysurface_create(surface, width, height);
|
2012-09-27 11:06:36 +02:00
|
|
|
qemu_alloc_display(surface, width, height, linesize,
|
|
|
|
qemu_default_pixelformat(32), 0);
|
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
2012-09-26 07:46:20 +02:00
|
|
|
DisplaySurface *qemu_create_displaysurface_from(int width, int height, int bpp,
|
2013-02-20 09:37:12 +01:00
|
|
|
int linesize, uint8_t *data,
|
|
|
|
bool byteswap)
|
2010-02-11 00:29:57 +01:00
|
|
|
{
|
2012-09-26 15:20:05 +02:00
|
|
|
DisplaySurface *surface = g_new0(DisplaySurface, 1);
|
2010-02-11 00:29:57 +01:00
|
|
|
|
2013-02-28 10:48:02 +01:00
|
|
|
trace_displaysurface_create_from(surface, width, height, bpp, byteswap);
|
2013-02-20 09:37:12 +01:00
|
|
|
if (byteswap) {
|
|
|
|
surface->pf = qemu_different_endianness_pixelformat(bpp);
|
|
|
|
} else {
|
|
|
|
surface->pf = qemu_default_pixelformat(bpp);
|
|
|
|
}
|
2012-09-26 15:20:05 +02:00
|
|
|
|
|
|
|
surface->format = qemu_pixman_get_format(&surface->pf);
|
|
|
|
assert(surface->format != 0);
|
|
|
|
surface->image = pixman_image_create_bits(surface->format,
|
|
|
|
width, height,
|
|
|
|
(void *)data, linesize);
|
|
|
|
assert(surface->image != NULL);
|
|
|
|
|
2010-02-11 00:29:57 +01:00
|
|
|
#ifdef HOST_WORDS_BIGENDIAN
|
|
|
|
surface->flags = QEMU_BIG_ENDIAN_FLAG;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
2013-02-28 10:48:02 +01:00
|
|
|
void qemu_free_displaysurface(DisplaySurface *surface)
|
2010-02-11 00:29:57 +01:00
|
|
|
{
|
2013-02-28 10:48:02 +01:00
|
|
|
if (surface == NULL) {
|
2010-02-11 00:29:57 +01:00
|
|
|
return;
|
2012-09-26 07:46:20 +02:00
|
|
|
}
|
2013-02-28 10:48:02 +01:00
|
|
|
trace_displaysurface_free(surface);
|
|
|
|
qemu_pixman_image_unref(surface->image);
|
|
|
|
g_free(surface);
|
2010-02-11 00:29:57 +01:00
|
|
|
}
|
|
|
|
|
2012-11-13 14:51:41 +01:00
|
|
|
void register_displaychangelistener(DisplayState *ds,
|
|
|
|
DisplayChangeListener *dcl)
|
|
|
|
{
|
|
|
|
trace_displaychangelistener_register(dcl, dcl->ops->dpy_name);
|
|
|
|
dcl->ds = ds;
|
|
|
|
QLIST_INSERT_HEAD(&ds->listeners, dcl, next);
|
|
|
|
gui_setup_refresh(ds);
|
2013-02-28 15:03:04 +01:00
|
|
|
if (dcl->ops->dpy_gfx_switch) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_gfx_switch(dcl, ds->surface);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void unregister_displaychangelistener(DisplayChangeListener *dcl)
|
|
|
|
{
|
|
|
|
DisplayState *ds = dcl->ds;
|
|
|
|
trace_displaychangelistener_unregister(dcl, dcl->ops->dpy_name);
|
|
|
|
QLIST_REMOVE(dcl, next);
|
|
|
|
gui_setup_refresh(ds);
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void dpy_gfx_update(QemuConsole *con, int x, int y, int w, int h)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
int width = pixman_image_get_width(s->surface->image);
|
|
|
|
int height = pixman_image_get_height(s->surface->image);
|
|
|
|
|
|
|
|
x = MAX(x, 0);
|
|
|
|
y = MAX(y, 0);
|
|
|
|
x = MIN(x, width);
|
|
|
|
y = MIN(y, height);
|
|
|
|
w = MIN(w, width - x);
|
|
|
|
h = MIN(h, height - y);
|
|
|
|
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_gfx_update) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_gfx_update(dcl, x, y, w, h);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void dpy_gfx_replace_surface(QemuConsole *con,
|
2013-02-28 10:48:02 +01:00
|
|
|
DisplaySurface *surface)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2013-02-28 10:48:02 +01:00
|
|
|
DisplaySurface *old_surface = s->surface;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
2013-02-28 10:48:02 +01:00
|
|
|
|
|
|
|
s->surface = surface;
|
2012-11-13 14:51:41 +01:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2013-02-28 15:03:04 +01:00
|
|
|
if (dcl->ops->dpy_gfx_switch) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_gfx_switch(dcl, surface);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
2013-02-28 10:48:02 +01:00
|
|
|
qemu_free_displaysurface(old_surface);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void dpy_refresh(DisplayState *s)
|
|
|
|
{
|
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_refresh) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_refresh(dcl);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void dpy_gfx_copy(QemuConsole *con, int src_x, int src_y,
|
|
|
|
int dst_x, int dst_y, int w, int h)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_gfx_copy) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_gfx_copy(dcl, src_x, src_y, dst_x, dst_y, w, h);
|
2012-11-13 14:51:41 +01:00
|
|
|
} else { /* TODO */
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_gfx_update(dcl, dst_x, dst_y, w, h);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void dpy_text_cursor(QemuConsole *con, int x, int y)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_text_cursor) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_text_cursor(dcl, x, y);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void dpy_text_update(QemuConsole *con, int x, int y, int w, int h)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_text_update) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_text_update(dcl, x, y, w, h);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void dpy_text_resize(QemuConsole *con, int w, int h)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_text_resize) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_text_resize(dcl, w, h);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void dpy_mouse_set(QemuConsole *con, int x, int y, int on)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_mouse_set) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_mouse_set(dcl, x, y, on);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void dpy_cursor_define(QemuConsole *con, QEMUCursor *cursor)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_cursor_define) {
|
2013-03-01 13:03:04 +01:00
|
|
|
dcl->ops->dpy_cursor_define(dcl, cursor);
|
2012-11-13 14:51:41 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
bool dpy_cursor_define_supported(QemuConsole *con)
|
2012-11-13 14:51:41 +01:00
|
|
|
{
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplayState *s = con->ds;
|
2012-11-13 14:51:41 +01:00
|
|
|
struct DisplayChangeListener *dcl;
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_cursor_define) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-02-11 00:29:57 +01:00
|
|
|
static void dumb_display_init(void)
|
|
|
|
{
|
2011-08-20 22:09:37 -05:00
|
|
|
DisplayState *ds = g_malloc0(sizeof(DisplayState));
|
2011-06-19 11:53:02 +02:00
|
|
|
int width = 640;
|
|
|
|
int height = 480;
|
|
|
|
|
|
|
|
if (is_fixedsize_console()) {
|
|
|
|
width = active_console->g_width;
|
|
|
|
height = active_console->g_height;
|
|
|
|
}
|
2013-03-05 15:24:14 +01:00
|
|
|
ds->surface = qemu_create_displaysurface(width, height);
|
2013-02-28 10:48:02 +01:00
|
|
|
|
2010-02-11 00:29:57 +01:00
|
|
|
register_displaystate(ds);
|
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************/
|
|
|
|
/* register display */
|
|
|
|
|
|
|
|
void register_displaystate(DisplayState *ds)
|
|
|
|
{
|
|
|
|
DisplayState **s;
|
|
|
|
s = &display_state;
|
|
|
|
while (*s != NULL)
|
|
|
|
s = &(*s)->next;
|
|
|
|
ds->next = NULL;
|
|
|
|
*s = ds;
|
|
|
|
}
|
|
|
|
|
|
|
|
DisplayState *get_displaystate(void)
|
|
|
|
{
|
|
|
|
if (!display_state) {
|
|
|
|
dumb_display_init ();
|
|
|
|
}
|
|
|
|
return display_state;
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
QemuConsole *graphic_console_init(vga_hw_update_ptr update,
|
|
|
|
vga_hw_invalidate_ptr invalidate,
|
|
|
|
vga_hw_screen_dump_ptr screen_dump,
|
|
|
|
vga_hw_text_update_ptr text_update,
|
|
|
|
void *opaque)
|
2006-04-09 01:06:34 +00:00
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s;
|
2009-01-16 19:04:14 +00:00
|
|
|
DisplayState *ds;
|
2009-01-16 21:13:49 +00:00
|
|
|
|
2011-08-20 22:09:37 -05:00
|
|
|
ds = (DisplayState *) g_malloc0(sizeof(DisplayState));
|
2007-07-11 23:14:59 +00:00
|
|
|
s = new_console(ds, GRAPHIC_CONSOLE);
|
2006-04-09 01:06:34 +00:00
|
|
|
s->hw_update = update;
|
|
|
|
s->hw_invalidate = invalidate;
|
|
|
|
s->hw_screen_dump = screen_dump;
|
2008-02-10 16:33:14 +00:00
|
|
|
s->hw_text_update = text_update;
|
2006-04-09 01:06:34 +00:00
|
|
|
s->hw = opaque;
|
2009-01-16 19:04:14 +00:00
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
ds->surface = qemu_create_displaysurface(640, 480);
|
2013-02-28 10:48:02 +01:00
|
|
|
|
2009-01-16 21:13:49 +00:00
|
|
|
register_displaystate(ds);
|
2013-03-05 15:24:14 +01:00
|
|
|
return s;
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
|
2006-04-09 01:06:34 +00:00
|
|
|
int is_graphic_console(void)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
2008-02-10 16:33:14 +00:00
|
|
|
return active_console && active_console->console_type == GRAPHIC_CONSOLE;
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
|
|
|
|
2008-09-24 03:32:33 +00:00
|
|
|
int is_fixedsize_console(void)
|
|
|
|
{
|
|
|
|
return active_console && active_console->console_type != TEXT_CONSOLE;
|
|
|
|
}
|
|
|
|
|
2010-12-23 13:42:52 +01:00
|
|
|
static void text_console_set_echo(CharDriverState *chr, bool echo)
|
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s = chr->opaque;
|
2010-12-23 13:42:52 +01:00
|
|
|
|
|
|
|
s->echo = echo;
|
|
|
|
}
|
|
|
|
|
2012-07-10 22:00:55 +02:00
|
|
|
static void text_console_update_cursor(void *opaque)
|
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s = opaque;
|
2012-07-10 22:00:55 +02:00
|
|
|
|
|
|
|
s->cursor_visible_phase = !s->cursor_visible_phase;
|
|
|
|
vga_hw_invalidate();
|
|
|
|
qemu_mod_timer(s->cursor_timer,
|
|
|
|
qemu_get_clock_ms(rt_clock) + CONSOLE_CURSOR_PERIOD / 2);
|
|
|
|
}
|
|
|
|
|
2010-12-23 13:42:53 +01:00
|
|
|
static void text_console_do_init(CharDriverState *chr, DisplayState *ds)
|
2004-07-14 17:28:59 +00:00
|
|
|
{
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s;
|
2006-03-11 15:35:30 +00:00
|
|
|
|
2010-12-23 13:42:51 +01:00
|
|
|
s = chr->opaque;
|
2009-09-10 10:58:49 +02:00
|
|
|
|
2004-07-14 17:28:59 +00:00
|
|
|
chr->chr_write = console_puts;
|
2004-08-01 21:48:30 +00:00
|
|
|
|
2006-06-25 16:26:29 +00:00
|
|
|
s->out_fifo.buf = s->out_fifo_buf;
|
|
|
|
s->out_fifo.buf_size = sizeof(s->out_fifo_buf);
|
2011-03-11 16:47:48 +01:00
|
|
|
s->kbd_timer = qemu_new_timer_ms(rt_clock, kbd_send_chars, s);
|
2009-01-16 19:04:14 +00:00
|
|
|
s->ds = ds;
|
2007-09-17 08:09:54 +00:00
|
|
|
|
2004-07-14 17:28:59 +00:00
|
|
|
s->y_displayed = 0;
|
|
|
|
s->y_base = 0;
|
|
|
|
s->total_height = DEFAULT_BACKSCROLL;
|
|
|
|
s->x = 0;
|
|
|
|
s->y = 0;
|
2010-12-23 13:42:51 +01:00
|
|
|
if (s->console_type == TEXT_CONSOLE) {
|
2013-03-06 13:40:47 +01:00
|
|
|
s->g_width = surface_width(s->ds->surface);
|
|
|
|
s->g_height = surface_height(s->ds->surface);
|
2010-12-23 13:42:51 +01:00
|
|
|
}
|
2006-03-11 15:35:30 +00:00
|
|
|
|
2012-07-10 22:00:55 +02:00
|
|
|
s->cursor_timer =
|
|
|
|
qemu_new_timer_ms(rt_clock, text_console_update_cursor, s);
|
|
|
|
|
2008-02-10 16:33:14 +00:00
|
|
|
s->hw_invalidate = text_console_invalidate;
|
|
|
|
s->hw_text_update = text_console_update;
|
|
|
|
s->hw = s;
|
|
|
|
|
2006-03-11 15:35:30 +00:00
|
|
|
/* Set text attribute defaults */
|
|
|
|
s->t_attrib_default.bold = 0;
|
|
|
|
s->t_attrib_default.uline = 0;
|
|
|
|
s->t_attrib_default.blink = 0;
|
|
|
|
s->t_attrib_default.invers = 0;
|
|
|
|
s->t_attrib_default.unvisible = 0;
|
|
|
|
s->t_attrib_default.fgcol = COLOR_WHITE;
|
|
|
|
s->t_attrib_default.bgcol = COLOR_BLACK;
|
|
|
|
/* set current text attributes to default */
|
|
|
|
s->t_attrib = s->t_attrib_default;
|
2004-07-14 17:28:59 +00:00
|
|
|
text_console_resize(s);
|
|
|
|
|
2009-12-08 13:11:39 +01:00
|
|
|
if (chr->label) {
|
|
|
|
char msg[128];
|
|
|
|
int len;
|
|
|
|
|
2009-12-08 13:11:40 +01:00
|
|
|
s->t_attrib.bgcol = COLOR_BLUE;
|
2009-12-08 13:11:39 +01:00
|
|
|
len = snprintf(msg, sizeof(msg), "%s console\r\n", chr->label);
|
|
|
|
console_puts(chr, (uint8_t*)msg, len);
|
2009-12-08 13:11:40 +01:00
|
|
|
s->t_attrib = s->t_attrib_default;
|
2009-12-08 13:11:39 +01:00
|
|
|
}
|
|
|
|
|
2009-11-03 19:59:56 +05:30
|
|
|
qemu_chr_generic_open(chr);
|
2009-01-18 14:08:04 +00:00
|
|
|
if (chr->init)
|
|
|
|
chr->init(chr);
|
2004-07-14 17:28:59 +00:00
|
|
|
}
|
2008-07-01 16:24:38 +00:00
|
|
|
|
2013-02-25 15:52:32 +01:00
|
|
|
static CharDriverState *text_console_init(ChardevVC *vc)
|
2009-01-16 20:23:27 +00:00
|
|
|
{
|
|
|
|
CharDriverState *chr;
|
2012-09-28 13:24:17 +02:00
|
|
|
QemuConsole *s;
|
2013-02-25 15:52:32 +01:00
|
|
|
unsigned width = 0;
|
|
|
|
unsigned height = 0;
|
2009-01-16 20:23:27 +00:00
|
|
|
|
2011-08-20 22:09:37 -05:00
|
|
|
chr = g_malloc0(sizeof(CharDriverState));
|
2009-01-16 20:23:27 +00:00
|
|
|
|
2013-02-25 15:52:32 +01:00
|
|
|
if (vc->has_width) {
|
|
|
|
width = vc->width;
|
|
|
|
} else if (vc->has_cols) {
|
|
|
|
width = vc->cols * FONT_WIDTH;
|
|
|
|
}
|
2010-12-23 13:42:51 +01:00
|
|
|
|
2013-02-25 15:52:32 +01:00
|
|
|
if (vc->has_height) {
|
|
|
|
height = vc->height;
|
|
|
|
} else if (vc->has_rows) {
|
|
|
|
height = vc->rows * FONT_HEIGHT;
|
|
|
|
}
|
2010-12-23 13:42:51 +01:00
|
|
|
|
|
|
|
if (width == 0 || height == 0) {
|
|
|
|
s = new_console(NULL, TEXT_CONSOLE);
|
|
|
|
} else {
|
|
|
|
s = new_console(NULL, TEXT_CONSOLE_FIXED_SIZE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!s) {
|
2011-10-02 18:53:09 +02:00
|
|
|
g_free(chr);
|
Revert "qemu-char: Print strerror message on failure" and deps
The commit's purpose is laudable:
The only way for chardev drivers to communicate an error was to
return a NULL pointer, which resulted in an error message that
said _that_ something went wrong, but not _why_.
It attempts to achieve it by changing the interface to return 0/-errno
and update qemu_chr_open_opts() to use strerror() to display a more
helpful error message. Unfortunately, it has serious flaws:
1. Backends "socket" and "udp" return bogus error codes, because
qemu_chr_open_socket() and qemu_chr_open_udp() assume that
unix_listen_opts(), unix_connect_opts(), inet_listen_opts(),
inet_connect_opts() and inet_dgram_opts() fail with errno set
appropriately. That assumption is wrong, and the commit turns
unspecific error messages into misleading error messages. For
instance:
$ qemu-system-x86_64 -nodefaults -vnc :0 -chardev socket,id=bar,host=xxx
inet_connect: host and/or port not specified
chardev: opening backend "socket" failed: No such file or directory
ENOENT is what happens to be in my errno when the backend returns
-errno. Let's put ERANGE there just for giggles:
$ qemu-system-x86_64 -nodefaults -vnc :0 -chardev socket,id=bar,host=xxx -drive if=none,iops=99999999999999999999
inet_connect: host and/or port not specified
chardev: opening backend "socket" failed: Numerical result out of range
Worse: when errno happens to be zero, return -errno erroneously
signals success, and qemu_chr_new_from_opts() dies dereferencing
uninitialized chr. I observe this with "-serial unix:".
2. All qemu_chr_open_opts() knows about the error is an errno error
code. That's simply not enough for a decent message. For instance,
when inet_dgram() can't resolve the parameter host, which errno code
should it use? What if it can't resolve parameter localaddr?
Clue: many backends already report errors in their open methods.
Let's revert the flawed commit along with its dependencies, and fix up
the silent error paths instead.
This reverts commit 6e1db57b2ac9025c2443c665a0d9e78748637b26.
Conflicts:
console.c
hw/baum.c
qemu-char.c
This reverts commit aad04cd024f0c59f0b96f032cde2e24eb3abba6d.
The parts of commit db418a0a "Add stdio char device on windows" that
depend on the reverted change fixed up.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2012-02-07 15:09:08 +01:00
|
|
|
return NULL;
|
2010-12-23 13:42:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
s->chr = chr;
|
|
|
|
s->g_width = width;
|
|
|
|
s->g_height = height;
|
|
|
|
chr->opaque = s;
|
2010-12-23 13:42:52 +01:00
|
|
|
chr->chr_set_echo = text_console_set_echo;
|
Revert "qemu-char: Print strerror message on failure" and deps
The commit's purpose is laudable:
The only way for chardev drivers to communicate an error was to
return a NULL pointer, which resulted in an error message that
said _that_ something went wrong, but not _why_.
It attempts to achieve it by changing the interface to return 0/-errno
and update qemu_chr_open_opts() to use strerror() to display a more
helpful error message. Unfortunately, it has serious flaws:
1. Backends "socket" and "udp" return bogus error codes, because
qemu_chr_open_socket() and qemu_chr_open_udp() assume that
unix_listen_opts(), unix_connect_opts(), inet_listen_opts(),
inet_connect_opts() and inet_dgram_opts() fail with errno set
appropriately. That assumption is wrong, and the commit turns
unspecific error messages into misleading error messages. For
instance:
$ qemu-system-x86_64 -nodefaults -vnc :0 -chardev socket,id=bar,host=xxx
inet_connect: host and/or port not specified
chardev: opening backend "socket" failed: No such file or directory
ENOENT is what happens to be in my errno when the backend returns
-errno. Let's put ERANGE there just for giggles:
$ qemu-system-x86_64 -nodefaults -vnc :0 -chardev socket,id=bar,host=xxx -drive if=none,iops=99999999999999999999
inet_connect: host and/or port not specified
chardev: opening backend "socket" failed: Numerical result out of range
Worse: when errno happens to be zero, return -errno erroneously
signals success, and qemu_chr_new_from_opts() dies dereferencing
uninitialized chr. I observe this with "-serial unix:".
2. All qemu_chr_open_opts() knows about the error is an errno error
code. That's simply not enough for a decent message. For instance,
when inet_dgram() can't resolve the parameter host, which errno code
should it use? What if it can't resolve parameter localaddr?
Clue: many backends already report errors in their open methods.
Let's revert the flawed commit along with its dependencies, and fix up
the silent error paths instead.
This reverts commit 6e1db57b2ac9025c2443c665a0d9e78748637b26.
Conflicts:
console.c
hw/baum.c
qemu-char.c
This reverts commit aad04cd024f0c59f0b96f032cde2e24eb3abba6d.
The parts of commit db418a0a "Add stdio char device on windows" that
depend on the reverted change fixed up.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Signed-off-by: Anthony Liguori <aliguori@us.ibm.com>
2012-02-07 15:09:08 +01:00
|
|
|
return chr;
|
2009-01-16 20:23:27 +00:00
|
|
|
}
|
|
|
|
|
2013-02-20 07:43:19 -06:00
|
|
|
static VcHandler *vc_handler = text_console_init;
|
|
|
|
|
2013-02-25 15:52:32 +01:00
|
|
|
CharDriverState *vc_init(ChardevVC *vc)
|
2013-02-20 07:43:19 -06:00
|
|
|
{
|
2013-02-25 15:52:32 +01:00
|
|
|
return vc_handler(vc);
|
2013-02-20 07:43:19 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void register_vc_handler(VcHandler *handler)
|
|
|
|
{
|
|
|
|
vc_handler = handler;
|
|
|
|
}
|
|
|
|
|
2009-01-16 20:23:27 +00:00
|
|
|
void text_consoles_set_display(DisplayState *ds)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2012-02-07 15:09:21 +01:00
|
|
|
for (i = 0; i < nb_consoles; i++) {
|
|
|
|
if (consoles[i]->console_type != GRAPHIC_CONSOLE) {
|
|
|
|
text_console_do_init(consoles[i]->chr, ds);
|
|
|
|
}
|
2009-01-16 20:23:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void qemu_console_resize(QemuConsole *s, int width, int height)
|
2008-07-01 16:24:38 +00:00
|
|
|
{
|
2009-01-16 19:04:14 +00:00
|
|
|
s->g_width = width;
|
|
|
|
s->g_height = height;
|
|
|
|
if (is_graphic_console()) {
|
2013-02-28 10:48:02 +01:00
|
|
|
DisplaySurface *surface;
|
|
|
|
surface = qemu_create_displaysurface(width, height);
|
2013-03-05 15:24:14 +01:00
|
|
|
dpy_gfx_replace_surface(s, surface);
|
2008-07-01 16:24:38 +00:00
|
|
|
}
|
|
|
|
}
|
2008-09-24 02:21:24 +00:00
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
void qemu_console_copy(QemuConsole *con, int src_x, int src_y,
|
2009-01-16 19:04:14 +00:00
|
|
|
int dst_x, int dst_y, int w, int h)
|
2008-09-24 03:32:33 +00:00
|
|
|
{
|
2009-01-16 19:04:14 +00:00
|
|
|
if (is_graphic_console()) {
|
2013-03-05 15:24:14 +01:00
|
|
|
dpy_gfx_copy(con, src_x, src_y, dst_x, dst_y, w, h);
|
2008-09-24 02:21:24 +00:00
|
|
|
}
|
|
|
|
}
|
2009-01-15 22:14:11 +00:00
|
|
|
|
2013-03-05 15:24:14 +01:00
|
|
|
DisplaySurface *qemu_console_surface(QemuConsole *console)
|
|
|
|
{
|
|
|
|
return console->ds->surface;
|
|
|
|
}
|
|
|
|
|
|
|
|
DisplayState *qemu_console_displaystate(QemuConsole *console)
|
|
|
|
{
|
|
|
|
return console->ds;
|
|
|
|
}
|
|
|
|
|
2009-01-23 19:56:19 +00:00
|
|
|
PixelFormat qemu_different_endianness_pixelformat(int bpp)
|
2009-01-15 22:14:11 +00:00
|
|
|
{
|
|
|
|
PixelFormat pf;
|
|
|
|
|
|
|
|
memset(&pf, 0x00, sizeof(PixelFormat));
|
|
|
|
|
|
|
|
pf.bits_per_pixel = bpp;
|
2012-08-22 17:19:42 +02:00
|
|
|
pf.bytes_per_pixel = DIV_ROUND_UP(bpp, 8);
|
2009-01-15 22:14:11 +00:00
|
|
|
pf.depth = bpp == 32 ? 24 : bpp;
|
|
|
|
|
|
|
|
switch (bpp) {
|
2009-01-23 19:56:19 +00:00
|
|
|
case 24:
|
|
|
|
pf.rmask = 0x000000FF;
|
|
|
|
pf.gmask = 0x0000FF00;
|
|
|
|
pf.bmask = 0x00FF0000;
|
|
|
|
pf.rmax = 255;
|
|
|
|
pf.gmax = 255;
|
|
|
|
pf.bmax = 255;
|
|
|
|
pf.rshift = 0;
|
|
|
|
pf.gshift = 8;
|
|
|
|
pf.bshift = 16;
|
2009-01-26 15:37:30 +00:00
|
|
|
pf.rbits = 8;
|
|
|
|
pf.gbits = 8;
|
|
|
|
pf.bbits = 8;
|
2009-01-15 22:14:11 +00:00
|
|
|
break;
|
2009-01-23 19:56:19 +00:00
|
|
|
case 32:
|
|
|
|
pf.rmask = 0x0000FF00;
|
|
|
|
pf.gmask = 0x00FF0000;
|
|
|
|
pf.bmask = 0xFF000000;
|
|
|
|
pf.amask = 0x00000000;
|
|
|
|
pf.amax = 255;
|
|
|
|
pf.rmax = 255;
|
|
|
|
pf.gmax = 255;
|
|
|
|
pf.bmax = 255;
|
|
|
|
pf.ashift = 0;
|
|
|
|
pf.rshift = 8;
|
|
|
|
pf.gshift = 16;
|
|
|
|
pf.bshift = 24;
|
2009-01-26 15:37:30 +00:00
|
|
|
pf.rbits = 8;
|
|
|
|
pf.gbits = 8;
|
|
|
|
pf.bbits = 8;
|
|
|
|
pf.abits = 8;
|
2009-01-23 19:56:19 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return pf;
|
|
|
|
}
|
|
|
|
|
|
|
|
PixelFormat qemu_default_pixelformat(int bpp)
|
|
|
|
{
|
|
|
|
PixelFormat pf;
|
|
|
|
|
|
|
|
memset(&pf, 0x00, sizeof(PixelFormat));
|
|
|
|
|
|
|
|
pf.bits_per_pixel = bpp;
|
2012-08-22 17:19:42 +02:00
|
|
|
pf.bytes_per_pixel = DIV_ROUND_UP(bpp, 8);
|
2009-01-23 19:56:19 +00:00
|
|
|
pf.depth = bpp == 32 ? 24 : bpp;
|
|
|
|
|
|
|
|
switch (bpp) {
|
2010-05-21 11:59:14 +02:00
|
|
|
case 15:
|
|
|
|
pf.bits_per_pixel = 16;
|
|
|
|
pf.rmask = 0x00007c00;
|
|
|
|
pf.gmask = 0x000003E0;
|
|
|
|
pf.bmask = 0x0000001F;
|
|
|
|
pf.rmax = 31;
|
|
|
|
pf.gmax = 31;
|
|
|
|
pf.bmax = 31;
|
|
|
|
pf.rshift = 10;
|
|
|
|
pf.gshift = 5;
|
|
|
|
pf.bshift = 0;
|
|
|
|
pf.rbits = 5;
|
|
|
|
pf.gbits = 5;
|
|
|
|
pf.bbits = 5;
|
|
|
|
break;
|
2009-01-15 22:14:11 +00:00
|
|
|
case 16:
|
|
|
|
pf.rmask = 0x0000F800;
|
|
|
|
pf.gmask = 0x000007E0;
|
|
|
|
pf.bmask = 0x0000001F;
|
|
|
|
pf.rmax = 31;
|
|
|
|
pf.gmax = 63;
|
|
|
|
pf.bmax = 31;
|
|
|
|
pf.rshift = 11;
|
|
|
|
pf.gshift = 5;
|
|
|
|
pf.bshift = 0;
|
2009-01-26 15:37:30 +00:00
|
|
|
pf.rbits = 5;
|
|
|
|
pf.gbits = 6;
|
|
|
|
pf.bbits = 5;
|
2009-01-15 22:14:11 +00:00
|
|
|
break;
|
|
|
|
case 24:
|
2009-01-23 19:56:19 +00:00
|
|
|
pf.rmask = 0x00FF0000;
|
|
|
|
pf.gmask = 0x0000FF00;
|
|
|
|
pf.bmask = 0x000000FF;
|
|
|
|
pf.rmax = 255;
|
|
|
|
pf.gmax = 255;
|
|
|
|
pf.bmax = 255;
|
|
|
|
pf.rshift = 16;
|
|
|
|
pf.gshift = 8;
|
|
|
|
pf.bshift = 0;
|
2009-01-26 15:37:30 +00:00
|
|
|
pf.rbits = 8;
|
|
|
|
pf.gbits = 8;
|
|
|
|
pf.bbits = 8;
|
2011-11-22 12:56:10 +01:00
|
|
|
break;
|
2009-01-15 22:14:11 +00:00
|
|
|
case 32:
|
|
|
|
pf.rmask = 0x00FF0000;
|
|
|
|
pf.gmask = 0x0000FF00;
|
|
|
|
pf.bmask = 0x000000FF;
|
|
|
|
pf.rmax = 255;
|
|
|
|
pf.gmax = 255;
|
|
|
|
pf.bmax = 255;
|
|
|
|
pf.rshift = 16;
|
|
|
|
pf.gshift = 8;
|
|
|
|
pf.bshift = 0;
|
2009-01-26 15:37:30 +00:00
|
|
|
pf.rbits = 8;
|
|
|
|
pf.gbits = 8;
|
|
|
|
pf.bbits = 8;
|
2009-01-15 22:14:11 +00:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return pf;
|
|
|
|
}
|
2013-03-05 23:21:32 +05:30
|
|
|
|
2013-02-25 15:52:32 +01:00
|
|
|
static void qemu_chr_parse_vc(QemuOpts *opts, ChardevBackend *backend,
|
|
|
|
Error **errp)
|
|
|
|
{
|
|
|
|
int val;
|
|
|
|
|
|
|
|
backend->vc = g_new0(ChardevVC, 1);
|
|
|
|
|
|
|
|
val = qemu_opt_get_number(opts, "width", 0);
|
|
|
|
if (val != 0) {
|
|
|
|
backend->vc->has_width = true;
|
|
|
|
backend->vc->width = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
val = qemu_opt_get_number(opts, "height", 0);
|
|
|
|
if (val != 0) {
|
|
|
|
backend->vc->has_height = true;
|
|
|
|
backend->vc->height = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
val = qemu_opt_get_number(opts, "cols", 0);
|
|
|
|
if (val != 0) {
|
|
|
|
backend->vc->has_cols = true;
|
|
|
|
backend->vc->cols = val;
|
|
|
|
}
|
|
|
|
|
|
|
|
val = qemu_opt_get_number(opts, "rows", 0);
|
|
|
|
if (val != 0) {
|
|
|
|
backend->vc->has_rows = true;
|
|
|
|
backend->vc->rows = val;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 23:21:32 +05:30
|
|
|
static void register_types(void)
|
|
|
|
{
|
2013-02-25 15:52:32 +01:00
|
|
|
register_char_driver_qapi("vc", CHARDEV_BACKEND_KIND_VC,
|
|
|
|
qemu_chr_parse_vc);
|
2013-03-05 23:21:32 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
type_init(register_types);
|