2004-07-15 01:28:59 +08:00
|
|
|
/*
|
|
|
|
* QEMU graphical console
|
2007-09-17 05:08:06 +08:00
|
|
|
*
|
2004-07-15 01:28:59 +08:00
|
|
|
* Copyright (c) 2004 Fabrice Bellard
|
2007-09-17 05:08:06 +08:00
|
|
|
*
|
2004-07-15 01:28:59 +08: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.
|
|
|
|
*/
|
2018-02-01 19:18:31 +08:00
|
|
|
|
2016-01-30 01:49:51 +08:00
|
|
|
#include "qemu/osdep.h"
|
2012-11-28 19:06:30 +08:00
|
|
|
#include "ui/console.h"
|
2013-04-17 16:21:27 +08:00
|
|
|
#include "hw/qdev-core.h"
|
2018-02-01 19:18:31 +08:00
|
|
|
#include "qapi/error.h"
|
2018-02-11 17:36:01 +08:00
|
|
|
#include "qapi/qapi-commands-ui.h"
|
2023-08-30 17:37:52 +08:00
|
|
|
#include "qapi/visitor.h"
|
2022-12-21 21:14:34 +08:00
|
|
|
#include "qemu/coroutine.h"
|
2021-09-17 03:22:36 +08:00
|
|
|
#include "qemu/fifo8.h"
|
2023-08-30 17:37:37 +08:00
|
|
|
#include "qemu/error-report.h"
|
2021-09-17 03:22:38 +08:00
|
|
|
#include "qemu/main-loop.h"
|
2019-05-23 22:35:07 +08:00
|
|
|
#include "qemu/module.h"
|
2018-02-01 19:18:46 +08:00
|
|
|
#include "qemu/option.h"
|
2012-12-18 01:20:00 +08:00
|
|
|
#include "qemu/timer.h"
|
2021-09-17 03:22:38 +08:00
|
|
|
#include "chardev/char.h"
|
2013-11-10 21:20:16 +08:00
|
|
|
#include "trace.h"
|
2014-06-19 14:46:08 +08:00
|
|
|
#include "exec/memory.h"
|
2020-09-04 04:43:22 +08:00
|
|
|
#include "qom/object.h"
|
2004-07-15 01:28:59 +08:00
|
|
|
|
|
|
|
#define DEFAULT_BACKSCROLL 512
|
2012-07-11 04:00:55 +08:00
|
|
|
#define CONSOLE_CURSOR_PERIOD 500
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2006-03-11 23:35:30 +08: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;
|
|
|
|
|
2023-08-30 17:37:47 +08:00
|
|
|
#define TEXT_ATTRIBUTES_DEFAULT ((TextAttributes) { \
|
|
|
|
.fgcol = QEMU_COLOR_WHITE, \
|
|
|
|
.bgcol = QEMU_COLOR_BLACK \
|
|
|
|
})
|
|
|
|
|
2004-07-15 01:28:59 +08:00
|
|
|
typedef struct TextCell {
|
|
|
|
uint8_t ch;
|
2006-03-11 23:35:30 +08:00
|
|
|
TextAttributes t_attrib;
|
2004-07-15 01:28:59 +08:00
|
|
|
} TextCell;
|
|
|
|
|
|
|
|
#define MAX_ESC_PARAMS 3
|
|
|
|
|
|
|
|
enum TTYState {
|
|
|
|
TTY_STATE_NORM,
|
|
|
|
TTY_STATE_ESC,
|
|
|
|
TTY_STATE_CSI,
|
|
|
|
};
|
|
|
|
|
2012-09-28 19:24:17 +08:00
|
|
|
struct QemuConsole {
|
2013-04-17 15:45:10 +08:00
|
|
|
Object parent;
|
|
|
|
|
2011-09-16 06:48:07 +08:00
|
|
|
int index;
|
2004-07-15 01:28:59 +08:00
|
|
|
DisplayState *ds;
|
2013-03-12 21:39:22 +08:00
|
|
|
DisplaySurface *surface;
|
2021-02-20 20:23:03 +08:00
|
|
|
DisplayScanout scanout;
|
2013-03-15 22:45:54 +08:00
|
|
|
int dcls;
|
2021-10-10 03:48:46 +08:00
|
|
|
DisplayGLCtx *gl;
|
2021-03-11 15:45:33 +08:00
|
|
|
int gl_block;
|
2021-03-11 15:56:58 +08:00
|
|
|
QEMUTimer *gl_unblock_timer;
|
2016-12-21 08:38:04 +08:00
|
|
|
int window_id;
|
2014-01-25 00:38:20 +08:00
|
|
|
QemuUIInfo ui_info;
|
2015-03-12 19:51:13 +08:00
|
|
|
QEMUTimer *ui_timer;
|
2013-03-13 21:04:18 +08:00
|
|
|
const GraphicHwOps *hw_ops;
|
2006-04-09 09:06:34 +08:00
|
|
|
void *hw;
|
2020-10-27 21:36:02 +08:00
|
|
|
CoQueue dump_queue;
|
2018-05-07 17:54:24 +08:00
|
|
|
|
|
|
|
QTAILQ_ENTRY(QemuConsole) next;
|
2004-07-15 01:28:59 +08:00
|
|
|
};
|
|
|
|
|
2023-08-30 17:37:54 +08:00
|
|
|
OBJECT_DEFINE_ABSTRACT_TYPE(QemuConsole, qemu_console, QEMU_CONSOLE, OBJECT)
|
2023-08-30 17:37:51 +08:00
|
|
|
|
2023-08-30 17:37:53 +08:00
|
|
|
typedef struct QemuGraphicConsole {
|
|
|
|
QemuConsole parent;
|
2023-08-30 17:38:03 +08:00
|
|
|
|
|
|
|
Object *device;
|
|
|
|
uint32_t head;
|
|
|
|
|
|
|
|
QEMUCursor *cursor;
|
|
|
|
int cursor_x, cursor_y, cursor_on;
|
2023-08-30 17:37:53 +08:00
|
|
|
} QemuGraphicConsole;
|
|
|
|
|
|
|
|
typedef QemuConsoleClass QemuGraphicConsoleClass;
|
|
|
|
|
|
|
|
OBJECT_DEFINE_TYPE(QemuGraphicConsole, qemu_graphic_console, QEMU_GRAPHIC_CONSOLE, QEMU_CONSOLE)
|
|
|
|
|
|
|
|
typedef struct QemuTextConsole {
|
|
|
|
QemuConsole parent;
|
2023-08-30 17:38:02 +08:00
|
|
|
|
|
|
|
int width;
|
|
|
|
int height;
|
|
|
|
int total_height;
|
|
|
|
int backscroll_height;
|
|
|
|
int x, y;
|
|
|
|
int y_displayed;
|
|
|
|
int y_base;
|
|
|
|
TextCell *cells;
|
|
|
|
int text_x[2], text_y[2], cursor_invalidate;
|
|
|
|
int echo;
|
|
|
|
|
|
|
|
int update_x0;
|
|
|
|
int update_y0;
|
|
|
|
int update_x1;
|
|
|
|
int update_y1;
|
|
|
|
|
|
|
|
Chardev *chr;
|
|
|
|
/* fifo for key pressed */
|
|
|
|
Fifo8 out_fifo;
|
2023-08-30 17:37:53 +08:00
|
|
|
} QemuTextConsole;
|
|
|
|
|
|
|
|
typedef QemuConsoleClass QemuTextConsoleClass;
|
|
|
|
|
|
|
|
OBJECT_DEFINE_TYPE(QemuTextConsole, qemu_text_console, QEMU_TEXT_CONSOLE, QEMU_CONSOLE)
|
|
|
|
|
|
|
|
typedef struct QemuFixedTextConsole {
|
|
|
|
QemuTextConsole parent;
|
|
|
|
} QemuFixedTextConsole;
|
|
|
|
|
|
|
|
typedef QemuTextConsoleClass QemuFixedTextConsoleClass;
|
|
|
|
|
|
|
|
OBJECT_DEFINE_TYPE(QemuFixedTextConsole, qemu_fixed_text_console, QEMU_FIXED_TEXT_CONSOLE, QEMU_TEXT_CONSOLE)
|
|
|
|
|
2023-08-30 17:37:46 +08:00
|
|
|
struct VCChardev {
|
|
|
|
Chardev parent;
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *console;
|
2023-08-30 17:37:50 +08:00
|
|
|
|
|
|
|
enum TTYState state;
|
|
|
|
int esc_params[MAX_ESC_PARAMS];
|
|
|
|
int nb_esc_params;
|
|
|
|
TextAttributes t_attrib; /* currently active text attributes */
|
|
|
|
int x_saved, y_saved;
|
2023-08-30 17:37:46 +08:00
|
|
|
};
|
|
|
|
typedef struct VCChardev VCChardev;
|
|
|
|
|
2013-03-13 19:25:25 +08:00
|
|
|
struct DisplayState {
|
2013-12-01 15:49:47 +08:00
|
|
|
QEMUTimer *gui_timer;
|
2013-03-14 18:56:16 +08:00
|
|
|
uint64_t last_update;
|
|
|
|
uint64_t update_interval;
|
|
|
|
bool refreshing;
|
2013-03-13 19:25:25 +08:00
|
|
|
|
|
|
|
QLIST_HEAD(, DisplayChangeListener) listeners;
|
|
|
|
};
|
|
|
|
|
2010-02-11 07:29:57 +08:00
|
|
|
static DisplayState *display_state;
|
2012-09-28 19:24:17 +08:00
|
|
|
static QemuConsole *active_console;
|
2018-12-06 20:10:34 +08:00
|
|
|
static QTAILQ_HEAD(, QemuConsole) consoles =
|
2018-05-07 17:54:24 +08:00
|
|
|
QTAILQ_HEAD_INITIALIZER(consoles);
|
2014-05-22 17:27:13 +08:00
|
|
|
static bool cursor_visible_phase;
|
|
|
|
static QEMUTimer *cursor_timer;
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2013-03-14 18:56:16 +08:00
|
|
|
static void dpy_refresh(DisplayState *s);
|
2013-04-23 21:44:31 +08:00
|
|
|
static DisplayState *get_alloc_displaystate(void);
|
2014-05-22 17:27:13 +08:00
|
|
|
static void text_console_update_cursor(void *opaque);
|
2021-02-20 20:23:03 +08:00
|
|
|
static bool displaychangelistener_has_dmabuf(DisplayChangeListener *dcl);
|
2022-02-17 00:16:55 +08:00
|
|
|
static bool console_compatible_with(QemuConsole *con,
|
|
|
|
DisplayChangeListener *dcl, Error **errp);
|
2023-08-30 17:37:57 +08:00
|
|
|
static QemuConsole *qemu_graphic_console_lookup_unused(void);
|
2023-08-30 17:37:59 +08:00
|
|
|
static void dpy_set_ui_info_timer(void *opaque);
|
2013-03-08 00:08:29 +08:00
|
|
|
|
2013-03-13 19:17:13 +08:00
|
|
|
static void gui_update(void *opaque)
|
|
|
|
{
|
2013-03-14 18:56:16 +08:00
|
|
|
uint64_t interval = GUI_REFRESH_INTERVAL_IDLE;
|
|
|
|
uint64_t dcl_interval;
|
2013-03-13 19:17:13 +08:00
|
|
|
DisplayState *ds = opaque;
|
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
|
2013-03-14 18:56:16 +08:00
|
|
|
ds->refreshing = true;
|
2013-03-13 19:17:13 +08:00
|
|
|
dpy_refresh(ds);
|
2013-03-14 18:56:16 +08:00
|
|
|
ds->refreshing = false;
|
2013-03-13 19:17:13 +08:00
|
|
|
|
|
|
|
QLIST_FOREACH(dcl, &ds->listeners, next) {
|
2013-03-14 18:56:16 +08:00
|
|
|
dcl_interval = dcl->update_interval ?
|
|
|
|
dcl->update_interval : GUI_REFRESH_INTERVAL_DEFAULT;
|
|
|
|
if (interval > dcl_interval) {
|
|
|
|
interval = dcl_interval;
|
2013-03-13 19:17:13 +08:00
|
|
|
}
|
|
|
|
}
|
2013-03-14 18:56:16 +08:00
|
|
|
if (ds->update_interval != interval) {
|
|
|
|
ds->update_interval = interval;
|
|
|
|
trace_console_refresh(interval);
|
|
|
|
}
|
2013-08-21 23:03:08 +08:00
|
|
|
ds->last_update = qemu_clock_get_ms(QEMU_CLOCK_REALTIME);
|
|
|
|
timer_mod(ds->gui_timer, ds->last_update + interval);
|
2013-03-13 19:17:13 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void gui_setup_refresh(DisplayState *ds)
|
|
|
|
{
|
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
bool need_timer = false;
|
|
|
|
|
|
|
|
QLIST_FOREACH(dcl, &ds->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_refresh != NULL) {
|
|
|
|
need_timer = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (need_timer && ds->gui_timer == NULL) {
|
2013-08-21 23:03:08 +08:00
|
|
|
ds->gui_timer = timer_new_ms(QEMU_CLOCK_REALTIME, gui_update, ds);
|
|
|
|
timer_mod(ds->gui_timer, qemu_clock_get_ms(QEMU_CLOCK_REALTIME));
|
2013-03-13 19:17:13 +08:00
|
|
|
}
|
|
|
|
if (!need_timer && ds->gui_timer != NULL) {
|
2013-08-21 23:03:08 +08:00
|
|
|
timer_free(ds->gui_timer);
|
2013-03-13 19:17:13 +08:00
|
|
|
ds->gui_timer = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-08-24 19:20:49 +08:00
|
|
|
void graphic_hw_update_done(QemuConsole *con)
|
|
|
|
{
|
2020-11-24 20:29:36 +08:00
|
|
|
if (con) {
|
2022-04-27 21:08:29 +08:00
|
|
|
qemu_co_enter_all(&con->dump_queue, NULL);
|
2020-11-24 20:29:36 +08:00
|
|
|
}
|
2015-08-24 19:20:49 +08:00
|
|
|
}
|
|
|
|
|
2013-03-12 20:44:38 +08:00
|
|
|
void graphic_hw_update(QemuConsole *con)
|
2006-04-09 09:06:34 +08:00
|
|
|
{
|
2015-08-24 19:20:49 +08:00
|
|
|
bool async = false;
|
2020-11-07 01:03:39 +08:00
|
|
|
con = con ? con : active_console;
|
2013-03-12 20:44:38 +08:00
|
|
|
if (!con) {
|
2020-11-07 01:03:39 +08:00
|
|
|
return;
|
2013-03-12 20:44:38 +08:00
|
|
|
}
|
2020-11-07 01:03:39 +08:00
|
|
|
if (con->hw_ops->gfx_update) {
|
2013-03-13 21:04:18 +08:00
|
|
|
con->hw_ops->gfx_update(con->hw);
|
2015-08-24 19:20:49 +08:00
|
|
|
async = con->hw_ops->gfx_update_async;
|
|
|
|
}
|
|
|
|
if (!async) {
|
|
|
|
graphic_hw_update_done(con);
|
2013-03-12 20:44:38 +08:00
|
|
|
}
|
2006-04-09 09:06:34 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:37 +08:00
|
|
|
static void graphic_hw_update_bh(void *con)
|
|
|
|
{
|
|
|
|
graphic_hw_update(con);
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_console_co_wait_update(QemuConsole *con)
|
|
|
|
{
|
|
|
|
if (qemu_co_queue_empty(&con->dump_queue)) {
|
|
|
|
/* Defer the update, it will restart the pending coroutines */
|
|
|
|
aio_bh_schedule_oneshot(qemu_get_aio_context(),
|
|
|
|
graphic_hw_update_bh, con);
|
|
|
|
}
|
|
|
|
qemu_co_queue_wait(&con->dump_queue, NULL);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2021-03-11 15:56:58 +08:00
|
|
|
static void graphic_hw_gl_unblock_timer(void *opaque)
|
|
|
|
{
|
|
|
|
warn_report("console: no gl-unblock within one second");
|
|
|
|
}
|
|
|
|
|
2015-12-03 19:34:25 +08:00
|
|
|
void graphic_hw_gl_block(QemuConsole *con, bool block)
|
|
|
|
{
|
2021-03-11 15:56:58 +08:00
|
|
|
uint64_t timeout;
|
2016-09-23 15:50:27 +08:00
|
|
|
assert(con != NULL);
|
|
|
|
|
2021-03-11 15:45:33 +08:00
|
|
|
if (block) {
|
|
|
|
con->gl_block++;
|
|
|
|
} else {
|
|
|
|
con->gl_block--;
|
2015-12-03 19:34:25 +08:00
|
|
|
}
|
2021-03-11 15:45:33 +08:00
|
|
|
assert(con->gl_block >= 0);
|
|
|
|
if (!con->hw_ops->gl_block) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if ((block && con->gl_block != 1) || (!block && con->gl_block != 0)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
con->hw_ops->gl_block(con->hw, block);
|
2021-03-11 15:56:58 +08:00
|
|
|
|
|
|
|
if (block) {
|
|
|
|
timeout = qemu_clock_get_ms(QEMU_CLOCK_REALTIME);
|
|
|
|
timeout += 1000; /* one sec */
|
|
|
|
timer_mod(con->gl_unblock_timer, timeout);
|
|
|
|
} else {
|
|
|
|
timer_del(con->gl_unblock_timer);
|
|
|
|
}
|
2015-12-03 19:34:25 +08:00
|
|
|
}
|
|
|
|
|
2016-12-21 08:38:04 +08:00
|
|
|
int qemu_console_get_window_id(QemuConsole *con)
|
|
|
|
{
|
|
|
|
return con->window_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_console_set_window_id(QemuConsole *con, int window_id)
|
|
|
|
{
|
|
|
|
con->window_id = window_id;
|
|
|
|
}
|
|
|
|
|
2013-03-12 20:44:38 +08:00
|
|
|
void graphic_hw_invalidate(QemuConsole *con)
|
2006-04-09 09:06:34 +08:00
|
|
|
{
|
2013-03-12 20:44:38 +08:00
|
|
|
if (!con) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2013-03-13 21:04:18 +08:00
|
|
|
if (con && con->hw_ops->invalidate) {
|
|
|
|
con->hw_ops->invalidate(con->hw);
|
2013-03-12 20:44:38 +08:00
|
|
|
}
|
2006-04-09 09:06:34 +08:00
|
|
|
}
|
|
|
|
|
2013-03-12 20:44:38 +08:00
|
|
|
void graphic_hw_text_update(QemuConsole *con, console_ch_t *chardata)
|
2008-02-11 00:33:14 +08:00
|
|
|
{
|
2013-03-12 20:44:38 +08:00
|
|
|
if (!con) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2013-03-13 21:04:18 +08:00
|
|
|
if (con && con->hw_ops->text_update) {
|
|
|
|
con->hw_ops->text_update(con->hw, chardata);
|
|
|
|
}
|
2008-02-11 00:33:14 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:11 +08:00
|
|
|
static void qemu_console_fill_rect(QemuConsole *con, int posx, int posy,
|
|
|
|
int width, int height, pixman_color_t color)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2013-03-06 20:40:47 +08:00
|
|
|
DisplaySurface *surface = qemu_console_surface(con);
|
2013-03-06 22:43:23 +08:00
|
|
|
pixman_rectangle16_t rect = {
|
|
|
|
.x = posx, .y = posy, .width = width, .height = height
|
|
|
|
};
|
|
|
|
|
2023-08-30 17:38:12 +08:00
|
|
|
assert(surface);
|
2013-03-06 22:43:23 +08:00
|
|
|
pixman_image_fill_rectangles(PIXMAN_OP_SRC, surface->image,
|
2013-03-08 15:43:24 +08:00
|
|
|
&color, 1, &rect);
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* copy from (xs, ys) to (xd, yd) a rectangle of size (w, h) */
|
2023-08-30 17:38:11 +08:00
|
|
|
static void qemu_console_bitblt(QemuConsole *con,
|
|
|
|
int xs, int ys, int xd, int yd, int w, int h)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2013-03-06 20:40:47 +08:00
|
|
|
DisplaySurface *surface = qemu_console_surface(con);
|
2013-03-06 22:43:23 +08:00
|
|
|
|
2023-08-30 17:38:12 +08:00
|
|
|
assert(surface);
|
2013-03-06 22:43:23 +08:00
|
|
|
pixman_image_composite(PIXMAN_OP_SRC,
|
|
|
|
surface->image, NULL, surface->image,
|
|
|
|
xs, ys, 0, 0, xd, yd, w, h);
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/***********************************************************/
|
|
|
|
/* basic char display */
|
|
|
|
|
|
|
|
#define FONT_HEIGHT 16
|
|
|
|
#define FONT_WIDTH 8
|
|
|
|
|
|
|
|
#include "vgafont.h"
|
|
|
|
|
2013-03-08 15:43:24 +08:00
|
|
|
static const pixman_color_t color_table_rgb[2][8] = {
|
2006-03-11 23:35:30 +08:00
|
|
|
{ /* dark */
|
2023-08-30 17:38:10 +08:00
|
|
|
[QEMU_COLOR_BLACK] = QEMU_PIXMAN_COLOR_BLACK,
|
|
|
|
[QEMU_COLOR_BLUE] = QEMU_PIXMAN_COLOR(0x00, 0x00, 0xaa), /* blue */
|
|
|
|
[QEMU_COLOR_GREEN] = QEMU_PIXMAN_COLOR(0x00, 0xaa, 0x00), /* green */
|
|
|
|
[QEMU_COLOR_CYAN] = QEMU_PIXMAN_COLOR(0x00, 0xaa, 0xaa), /* cyan */
|
|
|
|
[QEMU_COLOR_RED] = QEMU_PIXMAN_COLOR(0xaa, 0x00, 0x00), /* red */
|
|
|
|
[QEMU_COLOR_MAGENTA] = QEMU_PIXMAN_COLOR(0xaa, 0x00, 0xaa), /* magenta */
|
|
|
|
[QEMU_COLOR_YELLOW] = QEMU_PIXMAN_COLOR(0xaa, 0xaa, 0x00), /* yellow */
|
|
|
|
[QEMU_COLOR_WHITE] = QEMU_PIXMAN_COLOR_GRAY,
|
2006-03-11 23:35:30 +08:00
|
|
|
},
|
|
|
|
{ /* bright */
|
2023-08-30 17:38:10 +08:00
|
|
|
[QEMU_COLOR_BLACK] = QEMU_PIXMAN_COLOR_BLACK,
|
|
|
|
[QEMU_COLOR_BLUE] = QEMU_PIXMAN_COLOR(0x00, 0x00, 0xff), /* blue */
|
|
|
|
[QEMU_COLOR_GREEN] = QEMU_PIXMAN_COLOR(0x00, 0xff, 0x00), /* green */
|
|
|
|
[QEMU_COLOR_CYAN] = QEMU_PIXMAN_COLOR(0x00, 0xff, 0xff), /* cyan */
|
|
|
|
[QEMU_COLOR_RED] = QEMU_PIXMAN_COLOR(0xff, 0x00, 0x00), /* red */
|
|
|
|
[QEMU_COLOR_MAGENTA] = QEMU_PIXMAN_COLOR(0xff, 0x00, 0xff), /* magenta */
|
|
|
|
[QEMU_COLOR_YELLOW] = QEMU_PIXMAN_COLOR(0xff, 0xff, 0x00), /* yellow */
|
|
|
|
[QEMU_COLOR_WHITE] = QEMU_PIXMAN_COLOR(0xff, 0xff, 0xff), /* white */
|
2006-03-11 23:35:30 +08:00
|
|
|
}
|
2004-07-15 01:28:59 +08:00
|
|
|
};
|
|
|
|
|
2013-03-06 20:40:47 +08:00
|
|
|
static void vga_putcharxy(QemuConsole *s, int x, int y, int ch,
|
2006-03-11 23:35:30 +08:00
|
|
|
TextAttributes *t_attrib)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2013-03-06 22:44:10 +08:00
|
|
|
static pixman_image_t *glyphs[256];
|
2013-03-06 20:40:47 +08:00
|
|
|
DisplaySurface *surface = qemu_console_surface(s);
|
2013-03-08 15:43:24 +08:00
|
|
|
pixman_color_t fgcol, bgcol;
|
2006-03-11 23:35:30 +08:00
|
|
|
|
2023-08-30 17:38:12 +08:00
|
|
|
assert(surface);
|
2006-03-11 23:35:30 +08:00
|
|
|
if (t_attrib->invers) {
|
2013-03-06 16:50:51 +08:00
|
|
|
bgcol = color_table_rgb[t_attrib->bold][t_attrib->fgcol];
|
|
|
|
fgcol = color_table_rgb[t_attrib->bold][t_attrib->bgcol];
|
2006-03-11 23:35:30 +08:00
|
|
|
} else {
|
2013-03-06 16:50:51 +08:00
|
|
|
fgcol = color_table_rgb[t_attrib->bold][t_attrib->fgcol];
|
|
|
|
bgcol = color_table_rgb[t_attrib->bold][t_attrib->bgcol];
|
2006-03-11 23:35:30 +08:00
|
|
|
}
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2013-03-06 22:44:10 +08:00
|
|
|
if (!glyphs[ch]) {
|
|
|
|
glyphs[ch] = qemu_pixman_glyph_from_vgafont(FONT_HEIGHT, vgafont16, ch);
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
2013-03-06 22:44:10 +08:00
|
|
|
qemu_pixman_glyph_render(glyphs[ch], surface->image,
|
2013-03-08 15:43:24 +08:00
|
|
|
&fgcol, &bgcol, x, y, FONT_WIDTH, FONT_HEIGHT);
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
static void text_console_resize(QemuTextConsole *t)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuConsole *s = QEMU_CONSOLE(t);
|
2004-07-15 01:28:59 +08:00
|
|
|
TextCell *cells, *c, *c1;
|
2023-08-30 17:38:14 +08:00
|
|
|
int w1, x, y, last_width, w, h;
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2021-02-20 20:23:03 +08:00
|
|
|
assert(s->scanout.kind == SCANOUT_SURFACE);
|
|
|
|
|
2023-08-30 17:38:14 +08:00
|
|
|
w = surface_width(s->surface) / FONT_WIDTH;
|
|
|
|
h = surface_height(s->surface) / FONT_HEIGHT;
|
|
|
|
if (w == t->width && h == t->height) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
last_width = t->width;
|
2023-08-30 17:38:14 +08:00
|
|
|
t->width = w;
|
|
|
|
t->height = h;
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2023-08-30 17:38:15 +08:00
|
|
|
w1 = MIN(t->width, last_width);
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
cells = g_new(TextCell, t->width * t->total_height + 1);
|
2023-08-30 17:38:15 +08:00
|
|
|
for (y = 0; y < t->total_height; y++) {
|
2023-08-30 17:38:02 +08:00
|
|
|
c = &cells[y * t->width];
|
2004-07-15 01:28:59 +08:00
|
|
|
if (w1 > 0) {
|
2023-08-30 17:38:02 +08:00
|
|
|
c1 = &t->cells[y * last_width];
|
2023-08-30 17:38:15 +08:00
|
|
|
for (x = 0; x < w1; x++) {
|
2004-07-15 01:28:59 +08:00
|
|
|
*c++ = *c1++;
|
|
|
|
}
|
|
|
|
}
|
2023-08-30 17:38:15 +08:00
|
|
|
for (x = w1; x < t->width; x++) {
|
2004-07-15 01:28:59 +08:00
|
|
|
c->ch = ' ';
|
2023-08-30 17:37:47 +08:00
|
|
|
c->t_attrib = TEXT_ATTRIBUTES_DEFAULT;
|
2004-07-15 01:28:59 +08:00
|
|
|
c++;
|
|
|
|
}
|
|
|
|
}
|
2023-08-30 17:38:02 +08:00
|
|
|
g_free(t->cells);
|
|
|
|
t->cells = cells;
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
static void invalidate_xy(QemuTextConsole *s, int x, int y)
|
2009-01-21 11:02:52 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
if (!qemu_console_is_visible(QEMU_CONSOLE(s))) {
|
2014-05-22 17:01:42 +08:00
|
|
|
return;
|
|
|
|
}
|
2009-01-21 11:02:52 +08: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;
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:49 +08:00
|
|
|
static void vc_update_xy(VCChardev *vc, int x, int y)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = vc->console;
|
2004-07-15 01:28:59 +08:00
|
|
|
TextCell *c;
|
|
|
|
int y1, y2;
|
|
|
|
|
2023-08-30 17:37:48 +08: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);
|
2008-02-11 00:33:14 +08:00
|
|
|
|
2014-05-22 17:01:42 +08: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) {
|
2019-07-01 15:53:01 +08:00
|
|
|
if (x >= s->width) {
|
|
|
|
x = s->width - 1;
|
|
|
|
}
|
2014-05-22 17:01:42 +08:00
|
|
|
c = &s->cells[y1 * s->width + x];
|
2023-08-30 17:38:02 +08:00
|
|
|
vga_putcharxy(QEMU_CONSOLE(s), x, y2, c->ch,
|
2014-05-22 17:01:42 +08:00
|
|
|
&(c->t_attrib));
|
|
|
|
invalidate_xy(s, x, y2);
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
static void console_show_cursor(QemuTextConsole *s, int show)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
|
|
|
TextCell *c;
|
|
|
|
int y, y1;
|
2013-03-06 20:40:47 +08:00
|
|
|
int x = s->x;
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2023-08-30 17:37:39 +08:00
|
|
|
s->cursor_invalidate = 1;
|
2008-02-11 00:33:14 +08:00
|
|
|
|
2014-05-22 17:01:42 +08:00
|
|
|
if (x >= s->width) {
|
|
|
|
x = s->width - 1;
|
|
|
|
}
|
|
|
|
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) {
|
|
|
|
c = &s->cells[y1 * s->width + x];
|
2014-05-22 17:27:13 +08:00
|
|
|
if (show && cursor_visible_phase) {
|
2023-08-30 17:37:47 +08:00
|
|
|
TextAttributes t_attrib = TEXT_ATTRIBUTES_DEFAULT;
|
2014-05-22 17:01:42 +08:00
|
|
|
t_attrib.invers = !(t_attrib.invers); /* invert fg and bg */
|
2023-08-30 17:38:02 +08:00
|
|
|
vga_putcharxy(QEMU_CONSOLE(s), x, y, c->ch, &t_attrib);
|
2014-05-22 17:01:42 +08:00
|
|
|
} else {
|
2023-08-30 17:38:02 +08:00
|
|
|
vga_putcharxy(QEMU_CONSOLE(s), x, y, c->ch, &(c->t_attrib));
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
2014-05-22 17:01:42 +08:00
|
|
|
invalidate_xy(s, x, y);
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
static void console_refresh(QemuTextConsole *s)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
DisplaySurface *surface = qemu_console_surface(QEMU_CONSOLE(s));
|
2004-07-15 01:28:59 +08:00
|
|
|
TextCell *c;
|
|
|
|
int x, y, y1;
|
|
|
|
|
2023-08-30 17:38:12 +08:00
|
|
|
assert(surface);
|
2023-08-30 17:37:39 +08: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-15 01:28:59 +08:00
|
|
|
|
2023-08-30 17:38:11 +08:00
|
|
|
qemu_console_fill_rect(QEMU_CONSOLE(s), 0, 0, surface_width(surface), surface_height(surface),
|
|
|
|
color_table_rgb[0][QEMU_COLOR_BLACK]);
|
2014-05-22 17:01:42 +08: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++) {
|
2023-08-30 17:38:02 +08:00
|
|
|
vga_putcharxy(QEMU_CONSOLE(s), x, y, c->ch,
|
2014-05-22 17:01:42 +08:00
|
|
|
&(c->t_attrib));
|
|
|
|
c++;
|
|
|
|
}
|
|
|
|
if (++y1 == s->total_height) {
|
|
|
|
y1 = 0;
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
}
|
2014-05-22 17:01:42 +08:00
|
|
|
console_show_cursor(s, 1);
|
2023-08-30 17:38:02 +08:00
|
|
|
dpy_gfx_update(QEMU_CONSOLE(s), 0, 0,
|
2014-05-22 17:01:42 +08:00
|
|
|
surface_width(surface), surface_height(surface));
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
static void console_scroll(QemuTextConsole *s, int ydelta)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
|
|
|
int i, y1;
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2004-07-15 01:28:59 +08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:49 +08:00
|
|
|
static void vc_put_lf(VCChardev *vc)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = vc->console;
|
2004-07-15 01:28:59 +08:00
|
|
|
TextCell *c;
|
|
|
|
int x, y1;
|
|
|
|
|
|
|
|
s->y++;
|
|
|
|
if (s->y >= s->height) {
|
|
|
|
s->y = s->height - 1;
|
2006-03-11 23:35:30 +08:00
|
|
|
|
2004-07-15 01:28:59 +08: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 = ' ';
|
2023-08-30 17:37:47 +08:00
|
|
|
c->t_attrib = TEXT_ATTRIBUTES_DEFAULT;
|
2004-07-15 01:28:59 +08:00
|
|
|
c++;
|
|
|
|
}
|
2014-05-22 17:01:42 +08:00
|
|
|
if (s->y_displayed == s->y_base) {
|
2023-08-30 17:37:39 +08: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;
|
2008-02-11 00:33:14 +08:00
|
|
|
|
2023-08-30 17:38:11 +08:00
|
|
|
qemu_console_bitblt(QEMU_CONSOLE(s), 0, FONT_HEIGHT, 0, 0,
|
|
|
|
s->width * FONT_WIDTH,
|
|
|
|
(s->height - 1) * FONT_HEIGHT);
|
|
|
|
qemu_console_fill_rect(QEMU_CONSOLE(s), 0, (s->height - 1) * FONT_HEIGHT,
|
|
|
|
s->width * FONT_WIDTH, FONT_HEIGHT,
|
|
|
|
color_table_rgb[0][TEXT_ATTRIBUTES_DEFAULT.bgcol]);
|
2014-05-22 17:01:42 +08:00
|
|
|
s->update_x0 = 0;
|
|
|
|
s->update_y0 = 0;
|
|
|
|
s->update_x1 = s->width * FONT_WIDTH;
|
|
|
|
s->update_y1 = s->height * FONT_HEIGHT;
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2006-03-11 23:35:30 +08: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.
|
|
|
|
*/
|
2023-08-30 17:37:49 +08:00
|
|
|
static void vc_handle_escape(VCChardev *vc)
|
2006-03-11 23:35:30 +08:00
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2023-08-30 17:37:50 +08:00
|
|
|
for (i = 0; i < vc->nb_esc_params; i++) {
|
|
|
|
switch (vc->esc_params[i]) {
|
2006-03-11 23:35:30 +08:00
|
|
|
case 0: /* reset all console attributes to default */
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib = TEXT_ATTRIBUTES_DEFAULT;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 1:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bold = 1;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 4:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.uline = 1;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 5:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.blink = 1;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 7:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.invers = 1;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 8:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.unvisible = 1;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 22:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bold = 0;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 24:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.uline = 0;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 25:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.blink = 0;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 27:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.invers = 0;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 28:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.unvisible = 0;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
/* set foreground color */
|
|
|
|
case 30:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.fgcol = QEMU_COLOR_BLACK;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 31:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.fgcol = QEMU_COLOR_RED;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 32:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.fgcol = QEMU_COLOR_GREEN;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 33:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.fgcol = QEMU_COLOR_YELLOW;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 34:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.fgcol = QEMU_COLOR_BLUE;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 35:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.fgcol = QEMU_COLOR_MAGENTA;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 36:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.fgcol = QEMU_COLOR_CYAN;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 37:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.fgcol = QEMU_COLOR_WHITE;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
/* set background color */
|
|
|
|
case 40:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bgcol = QEMU_COLOR_BLACK;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 41:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bgcol = QEMU_COLOR_RED;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 42:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bgcol = QEMU_COLOR_GREEN;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 43:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bgcol = QEMU_COLOR_YELLOW;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 44:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bgcol = QEMU_COLOR_BLUE;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 45:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bgcol = QEMU_COLOR_MAGENTA;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 46:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bgcol = QEMU_COLOR_CYAN;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case 47:
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->t_attrib.bgcol = QEMU_COLOR_WHITE;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:49 +08:00
|
|
|
static void vc_clear_xy(VCChardev *vc, int x, int y)
|
2007-01-17 07:02:36 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = vc->console;
|
2007-01-17 07:02:36 +08:00
|
|
|
int y1 = (s->y_base + y) % s->total_height;
|
2019-07-01 15:53:01 +08:00
|
|
|
if (x >= s->width) {
|
|
|
|
x = s->width - 1;
|
|
|
|
}
|
2007-01-17 07:02:36 +08:00
|
|
|
TextCell *c = &s->cells[y1 * s->width + x];
|
|
|
|
c->ch = ' ';
|
2023-08-30 17:37:47 +08:00
|
|
|
c->t_attrib = TEXT_ATTRIBUTES_DEFAULT;
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_update_xy(vc, x, y);
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:49 +08:00
|
|
|
static void vc_put_one(VCChardev *vc, int ch)
|
2016-03-09 03:51:21 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = vc->console;
|
2016-03-09 03:51:21 +08:00
|
|
|
TextCell *c;
|
|
|
|
int y1;
|
|
|
|
if (s->x >= s->width) {
|
|
|
|
/* line wrap */
|
|
|
|
s->x = 0;
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_put_lf(vc);
|
2016-03-09 03:51:21 +08:00
|
|
|
}
|
|
|
|
y1 = (s->y_base + s->y) % s->total_height;
|
|
|
|
c = &s->cells[y1 * s->width + s->x];
|
|
|
|
c->ch = ch;
|
2023-08-30 17:37:50 +08:00
|
|
|
c->t_attrib = vc->t_attrib;
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_update_xy(vc, s->x, s->y);
|
2016-03-09 03:51:21 +08:00
|
|
|
s->x++;
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:49 +08:00
|
|
|
static void vc_respond_str(VCChardev *vc, const char *buf)
|
2016-03-09 03:51:21 +08:00
|
|
|
{
|
|
|
|
while (*buf) {
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_put_one(vc, *buf);
|
2016-03-09 03:51:21 +08:00
|
|
|
buf++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-09-04 23:26:09 +08:00
|
|
|
/* set cursor, checking bounds */
|
2023-08-30 17:37:49 +08:00
|
|
|
static void vc_set_cursor(VCChardev *vc, int x, int y)
|
2012-09-04 23:26:09 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = vc->console;
|
2023-08-30 17:37:49 +08:00
|
|
|
|
2012-09-04 23:26:09 +08: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;
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:49 +08:00
|
|
|
static void vc_putchar(VCChardev *vc, int ch)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = vc->console;
|
2016-03-09 03:51:21 +08:00
|
|
|
int i;
|
2007-01-17 07:02:36 +08:00
|
|
|
int x, y;
|
2016-03-09 03:51:21 +08:00
|
|
|
char response[40];
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2023-08-30 17:37:50 +08:00
|
|
|
switch(vc->state) {
|
2004-07-15 01:28:59 +08:00
|
|
|
case TTY_STATE_NORM:
|
|
|
|
switch(ch) {
|
2006-03-11 23:35:30 +08:00
|
|
|
case '\r': /* carriage return */
|
2004-07-15 01:28:59 +08:00
|
|
|
s->x = 0;
|
|
|
|
break;
|
2006-03-11 23:35:30 +08:00
|
|
|
case '\n': /* newline */
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_put_lf(vc);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
2006-03-11 23:35:30 +08:00
|
|
|
case '\b': /* backspace */
|
2007-09-17 05:08:06 +08:00
|
|
|
if (s->x > 0)
|
2006-06-26 00:26:29 +08:00
|
|
|
s->x--;
|
2006-03-11 23:35:30 +08:00
|
|
|
break;
|
|
|
|
case '\t': /* tabspace */
|
|
|
|
if (s->x + (8 - (s->x % 8)) > s->width) {
|
2006-07-15 04:24:31 +08:00
|
|
|
s->x = 0;
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_put_lf(vc);
|
2006-03-11 23:35:30 +08:00
|
|
|
} else {
|
|
|
|
s->x = s->x + (8 - (s->x % 8));
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case '\a': /* alert aka. bell */
|
|
|
|
/* TODO: has to be implemented */
|
|
|
|
break;
|
2007-01-17 07:02:36 +08:00
|
|
|
case 14:
|
|
|
|
/* SI (shift in), character set 0 (ignored) */
|
|
|
|
break;
|
|
|
|
case 15:
|
|
|
|
/* SO (shift out), character set 1 (ignored) */
|
|
|
|
break;
|
2006-03-11 23:35:30 +08:00
|
|
|
case 27: /* esc (introducing an escape sequence) */
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->state = TTY_STATE_ESC;
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
|
|
|
default:
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_put_one(vc, ch);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2006-03-11 23:35:30 +08:00
|
|
|
case TTY_STATE_ESC: /* check if it is a terminal escape sequence */
|
2004-07-15 01:28:59 +08:00
|
|
|
if (ch == '[') {
|
|
|
|
for(i=0;i<MAX_ESC_PARAMS;i++)
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->esc_params[i] = 0;
|
|
|
|
vc->nb_esc_params = 0;
|
|
|
|
vc->state = TTY_STATE_CSI;
|
2004-07-15 01:28:59 +08:00
|
|
|
} else {
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->state = TTY_STATE_NORM;
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
break;
|
2006-03-11 23:35:30 +08:00
|
|
|
case TTY_STATE_CSI: /* handle escape sequence parameters */
|
2004-07-15 01:28:59 +08:00
|
|
|
if (ch >= '0' && ch <= '9') {
|
2023-08-30 17:37:50 +08:00
|
|
|
if (vc->nb_esc_params < MAX_ESC_PARAMS) {
|
|
|
|
int *param = &vc->esc_params[vc->nb_esc_params];
|
2012-09-17 17:10:03 +08:00
|
|
|
int digit = (ch - '0');
|
|
|
|
|
|
|
|
*param = (*param <= (INT_MAX - digit) / 10) ?
|
|
|
|
*param * 10 + digit : INT_MAX;
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
} else {
|
2023-08-30 17:37:50 +08:00
|
|
|
if (vc->nb_esc_params < MAX_ESC_PARAMS)
|
|
|
|
vc->nb_esc_params++;
|
2017-08-29 19:38:18 +08:00
|
|
|
if (ch == ';' || ch == '?') {
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
2017-08-29 19:38:18 +08:00
|
|
|
}
|
2023-08-30 17:37:50 +08:00
|
|
|
trace_console_putchar_csi(vc->esc_params[0], vc->esc_params[1],
|
|
|
|
ch, vc->nb_esc_params);
|
|
|
|
vc->state = TTY_STATE_NORM;
|
2004-07-15 01:28:59 +08:00
|
|
|
switch(ch) {
|
2007-01-17 07:02:36 +08:00
|
|
|
case 'A':
|
|
|
|
/* move cursor up */
|
2023-08-30 17:37:50 +08:00
|
|
|
if (vc->esc_params[0] == 0) {
|
|
|
|
vc->esc_params[0] = 1;
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
2023-08-30 17:37:50 +08:00
|
|
|
vc_set_cursor(vc, s->x, s->y - vc->esc_params[0]);
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
case 'B':
|
|
|
|
/* move cursor down */
|
2023-08-30 17:37:50 +08:00
|
|
|
if (vc->esc_params[0] == 0) {
|
|
|
|
vc->esc_params[0] = 1;
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
2023-08-30 17:37:50 +08:00
|
|
|
vc_set_cursor(vc, s->x, s->y + vc->esc_params[0]);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
|
|
|
case 'C':
|
2007-01-17 07:02:36 +08:00
|
|
|
/* move cursor right */
|
2023-08-30 17:37:50 +08:00
|
|
|
if (vc->esc_params[0] == 0) {
|
|
|
|
vc->esc_params[0] = 1;
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
2023-08-30 17:37:50 +08:00
|
|
|
vc_set_cursor(vc, s->x + vc->esc_params[0], s->y);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
2007-01-17 07:02:36 +08:00
|
|
|
case 'D':
|
|
|
|
/* move cursor left */
|
2023-08-30 17:37:50 +08:00
|
|
|
if (vc->esc_params[0] == 0) {
|
|
|
|
vc->esc_params[0] = 1;
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
2023-08-30 17:37:50 +08:00
|
|
|
vc_set_cursor(vc, s->x - vc->esc_params[0], s->y);
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
case 'G':
|
|
|
|
/* move cursor to column */
|
2023-08-30 17:37:50 +08:00
|
|
|
vc_set_cursor(vc, vc->esc_params[0] - 1, s->y);
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
case 'f':
|
|
|
|
case 'H':
|
|
|
|
/* move cursor to row, column */
|
2023-08-30 17:37:50 +08:00
|
|
|
vc_set_cursor(vc, vc->esc_params[1] - 1, vc->esc_params[0] - 1);
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
case 'J':
|
2023-08-30 17:37:50 +08:00
|
|
|
switch (vc->esc_params[0]) {
|
2007-01-17 07:02:36 +08:00
|
|
|
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;
|
|
|
|
}
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_clear_xy(vc, x, y);
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
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;
|
|
|
|
}
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_clear_xy(vc, x, y);
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* clear entire screen */
|
|
|
|
for (y = 0; y <= s->height; y++) {
|
|
|
|
for (x = 0; x < s->width; x++) {
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_clear_xy(vc, x, y);
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
|
|
|
}
|
2011-11-22 18:59:06 +08:00
|
|
|
break;
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
2011-11-22 18:59:07 +08:00
|
|
|
break;
|
2004-07-15 01:28:59 +08:00
|
|
|
case 'K':
|
2023-08-30 17:37:50 +08:00
|
|
|
switch (vc->esc_params[0]) {
|
2007-01-17 07:02:36 +08:00
|
|
|
case 0:
|
2011-11-22 18:59:06 +08:00
|
|
|
/* clear to eol */
|
|
|
|
for(x = s->x; x < s->width; x++) {
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_clear_xy(vc, x, s->y);
|
2011-11-22 18:59:06 +08:00
|
|
|
}
|
|
|
|
break;
|
2007-01-17 07:02:36 +08:00
|
|
|
case 1:
|
|
|
|
/* clear from beginning of line */
|
2019-07-01 15:53:01 +08:00
|
|
|
for (x = 0; x <= s->x && x < s->width; x++) {
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_clear_xy(vc, x, s->y);
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
/* clear entire line */
|
|
|
|
for(x = 0; x < s->width; x++) {
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_clear_xy(vc, x, s->y);
|
2007-01-17 07:02:36 +08:00
|
|
|
}
|
2011-11-22 18:59:06 +08:00
|
|
|
break;
|
|
|
|
}
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
case 'm':
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_handle_escape(vc);
|
2011-11-22 18:59:06 +08:00
|
|
|
break;
|
2007-01-17 07:02:36 +08:00
|
|
|
case 'n':
|
2023-08-30 17:37:50 +08:00
|
|
|
switch (vc->esc_params[0]) {
|
2016-03-09 03:51:21 +08:00
|
|
|
case 5:
|
|
|
|
/* report console status (always succeed)*/
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_respond_str(vc, "\033[0n");
|
2016-03-09 03:51:21 +08:00
|
|
|
break;
|
|
|
|
case 6:
|
|
|
|
/* report cursor position */
|
|
|
|
sprintf(response, "\033[%d;%dR",
|
|
|
|
(s->y_base + s->y) % s->total_height + 1,
|
|
|
|
s->x + 1);
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_respond_str(vc, response);
|
2016-03-09 03:51:21 +08:00
|
|
|
break;
|
|
|
|
}
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
/* save cursor position */
|
2023-08-30 17:37:50 +08:00
|
|
|
vc->x_saved = s->x;
|
|
|
|
vc->y_saved = s->y;
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
case 'u':
|
|
|
|
/* restore cursor position */
|
2023-08-30 17:37:50 +08:00
|
|
|
s->x = vc->x_saved;
|
|
|
|
s->y = vc->y_saved;
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
default:
|
2013-11-10 23:04:16 +08:00
|
|
|
trace_console_putchar_unhandled(ch);
|
2007-01-17 07:02:36 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
break;
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-17 16:42:20 +08:00
|
|
|
static void displaychangelistener_gfx_switch(DisplayChangeListener *dcl,
|
2022-02-21 03:31:58 +08:00
|
|
|
struct DisplaySurface *new_surface,
|
|
|
|
bool update)
|
2022-02-17 16:42:20 +08:00
|
|
|
{
|
|
|
|
if (dcl->ops->dpy_gfx_switch) {
|
|
|
|
dcl->ops->dpy_gfx_switch(dcl, new_surface);
|
|
|
|
}
|
2022-02-21 03:31:58 +08:00
|
|
|
|
|
|
|
if (update && dcl->ops->dpy_gfx_update) {
|
|
|
|
dcl->ops->dpy_gfx_update(dcl, 0, 0,
|
|
|
|
surface_width(new_surface),
|
|
|
|
surface_height(new_surface));
|
|
|
|
}
|
2022-02-17 16:42:20 +08:00
|
|
|
}
|
|
|
|
|
2022-02-17 19:07:21 +08:00
|
|
|
static void dpy_gfx_create_texture(QemuConsole *con, DisplaySurface *surface)
|
|
|
|
{
|
|
|
|
if (con->gl && con->gl->ops->dpy_gl_ctx_create_texture) {
|
|
|
|
con->gl->ops->dpy_gl_ctx_create_texture(con->gl, surface);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dpy_gfx_destroy_texture(QemuConsole *con, DisplaySurface *surface)
|
|
|
|
{
|
|
|
|
if (con->gl && con->gl->ops->dpy_gl_ctx_destroy_texture) {
|
|
|
|
con->gl->ops->dpy_gl_ctx_destroy_texture(con->gl, surface);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void dpy_gfx_update_texture(QemuConsole *con, DisplaySurface *surface,
|
|
|
|
int x, int y, int w, int h)
|
|
|
|
{
|
|
|
|
if (con->gl && con->gl->ops->dpy_gl_ctx_update_texture) {
|
|
|
|
con->gl->ops->dpy_gl_ctx_update_texture(con->gl, surface, x, y, w, h);
|
|
|
|
}
|
|
|
|
}
|
2022-02-17 16:42:20 +08:00
|
|
|
|
2021-02-20 20:23:03 +08:00
|
|
|
static void displaychangelistener_display_console(DisplayChangeListener *dcl,
|
2022-02-17 00:16:55 +08:00
|
|
|
QemuConsole *con,
|
|
|
|
Error **errp)
|
2021-02-20 20:23:03 +08:00
|
|
|
{
|
|
|
|
static const char nodev[] =
|
|
|
|
"This VM has no graphic display device.";
|
|
|
|
static DisplaySurface *dummy;
|
|
|
|
|
2022-02-17 00:16:55 +08:00
|
|
|
if (!con || !console_compatible_with(con, dcl, errp)) {
|
2021-02-20 20:23:03 +08:00
|
|
|
if (!dummy) {
|
|
|
|
dummy = qemu_create_placeholder_surface(640, 480, nodev);
|
|
|
|
}
|
2022-02-17 19:07:21 +08:00
|
|
|
if (con) {
|
|
|
|
dpy_gfx_create_texture(con, dummy);
|
|
|
|
}
|
2022-02-21 03:31:58 +08:00
|
|
|
displaychangelistener_gfx_switch(dcl, dummy, TRUE);
|
2021-02-20 20:23:03 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-02-21 03:45:38 +08:00
|
|
|
dpy_gfx_create_texture(con, con->surface);
|
|
|
|
displaychangelistener_gfx_switch(dcl, con->surface,
|
|
|
|
con->scanout.kind == SCANOUT_SURFACE);
|
|
|
|
|
2021-02-20 20:23:03 +08:00
|
|
|
if (con->scanout.kind == SCANOUT_DMABUF &&
|
|
|
|
displaychangelistener_has_dmabuf(dcl)) {
|
|
|
|
dcl->ops->dpy_gl_scanout_dmabuf(dcl, con->scanout.dmabuf);
|
|
|
|
} else if (con->scanout.kind == SCANOUT_TEXTURE &&
|
|
|
|
dcl->ops->dpy_gl_scanout_texture) {
|
|
|
|
dcl->ops->dpy_gl_scanout_texture(dcl,
|
|
|
|
con->scanout.texture.backing_id,
|
|
|
|
con->scanout.texture.backing_y_0_top,
|
|
|
|
con->scanout.texture.backing_width,
|
|
|
|
con->scanout.texture.backing_height,
|
|
|
|
con->scanout.texture.x,
|
|
|
|
con->scanout.texture.y,
|
|
|
|
con->scanout.texture.width,
|
2023-06-06 19:56:56 +08:00
|
|
|
con->scanout.texture.height,
|
|
|
|
con->scanout.texture.d3d_tex2d);
|
2021-02-20 20:23:03 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-07-15 01:28:59 +08:00
|
|
|
void console_select(unsigned int index)
|
|
|
|
{
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2012-09-28 19:24:17 +08:00
|
|
|
QemuConsole *s;
|
2006-03-11 23:35:30 +08:00
|
|
|
|
2013-03-07 23:04:52 +08:00
|
|
|
trace_console_select(index);
|
2013-03-15 22:45:54 +08:00
|
|
|
s = qemu_console_lookup_by_index(index);
|
2004-07-15 01:28:59 +08:00
|
|
|
if (s) {
|
2009-01-16 06:14:11 +08:00
|
|
|
DisplayState *ds = s->ds;
|
2012-07-11 04:00:55 +08:00
|
|
|
|
2004-07-15 01:28:59 +08:00
|
|
|
active_console = s;
|
2023-08-30 17:37:40 +08:00
|
|
|
QLIST_FOREACH (dcl, &ds->listeners, next) {
|
|
|
|
if (dcl->con != NULL) {
|
|
|
|
continue;
|
2018-03-09 00:18:03 +08:00
|
|
|
}
|
2023-08-30 17:37:40 +08:00
|
|
|
displaychangelistener_display_console(dcl, s, NULL);
|
2012-09-28 21:02:08 +08:00
|
|
|
}
|
2023-08-30 17:38:02 +08:00
|
|
|
|
|
|
|
if (QEMU_IS_TEXT_CONSOLE(s)) {
|
|
|
|
dpy_text_resize(s, QEMU_TEXT_CONSOLE(s)->width, QEMU_TEXT_CONSOLE(s)->height);
|
|
|
|
text_console_update_cursor(NULL);
|
|
|
|
}
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-07 23:39:10 +08:00
|
|
|
#define TYPE_CHARDEV_VC "chardev-vc"
|
2020-09-01 05:07:33 +08:00
|
|
|
DECLARE_INSTANCE_CHECKER(VCChardev, VC_CHARDEV,
|
|
|
|
TYPE_CHARDEV_VC)
|
2016-12-07 23:39:10 +08:00
|
|
|
|
2017-01-06 00:30:29 +08:00
|
|
|
static int vc_chr_write(Chardev *chr, const uint8_t *buf, int len)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2016-12-07 23:39:10 +08:00
|
|
|
VCChardev *drv = VC_CHARDEV(chr);
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = drv->console;
|
2004-07-15 01:28:59 +08:00
|
|
|
int i;
|
|
|
|
|
2009-01-21 11:02:52 +08: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-15 01:28:59 +08:00
|
|
|
console_show_cursor(s, 0);
|
|
|
|
for(i = 0; i < len; i++) {
|
2023-08-30 17:37:49 +08:00
|
|
|
vc_putchar(drv, buf[i]);
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
console_show_cursor(s, 1);
|
2023-08-30 17:37:41 +08:00
|
|
|
if (s->update_x0 < s->update_x1) {
|
2023-08-30 17:38:02 +08:00
|
|
|
dpy_gfx_update(QEMU_CONSOLE(s), s->update_x0, s->update_y0,
|
2012-09-28 21:02:08 +08:00
|
|
|
s->update_x1 - s->update_x0,
|
|
|
|
s->update_y1 - s->update_y0);
|
2009-01-21 11:02:52 +08:00
|
|
|
}
|
2004-07-15 01:28:59 +08:00
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
static void kbd_send_chars(QemuTextConsole *s)
|
2006-06-26 00:26:29 +08:00
|
|
|
{
|
2021-09-17 03:22:36 +08:00
|
|
|
uint32_t len, avail;
|
2007-09-17 16:09:54 +08:00
|
|
|
|
2011-08-16 00:17:31 +08:00
|
|
|
len = qemu_chr_be_can_write(s->chr);
|
2021-09-17 03:22:36 +08:00
|
|
|
avail = fifo8_num_used(&s->out_fifo);
|
2021-09-17 03:22:37 +08:00
|
|
|
while (len > 0 && avail > 0) {
|
2021-09-17 03:22:36 +08:00
|
|
|
const uint8_t *buf;
|
|
|
|
uint32_t size;
|
|
|
|
|
2021-09-17 03:22:37 +08:00
|
|
|
buf = fifo8_pop_buf(&s->out_fifo, MIN(len, avail), &size);
|
2022-10-22 22:12:04 +08:00
|
|
|
qemu_chr_be_write(s->chr, buf, size);
|
2021-09-17 03:22:37 +08:00
|
|
|
len = qemu_chr_be_can_write(s->chr);
|
2021-09-17 03:22:36 +08:00
|
|
|
avail -= size;
|
2006-06-26 00:26:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-07-15 01:28:59 +08:00
|
|
|
/* called when an ascii key is pressed */
|
2023-08-30 17:38:02 +08:00
|
|
|
void kbd_put_keysym_console(QemuConsole *con, int keysym)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = (QemuTextConsole *)object_dynamic_cast(OBJECT(con), TYPE_QEMU_TEXT_CONSOLE);
|
2004-07-15 01:28:59 +08:00
|
|
|
uint8_t buf[16], *q;
|
|
|
|
int c;
|
2021-09-17 03:22:36 +08:00
|
|
|
uint32_t num_free;
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
if (!s)
|
2004-07-15 01:28:59 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
switch(keysym) {
|
|
|
|
case QEMU_KEY_CTRL_UP:
|
2013-03-14 21:27:08 +08:00
|
|
|
console_scroll(s, -1);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
|
|
|
case QEMU_KEY_CTRL_DOWN:
|
2013-03-14 21:27:08 +08:00
|
|
|
console_scroll(s, 1);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
|
|
|
case QEMU_KEY_CTRL_PAGEUP:
|
2013-03-14 21:27:08 +08:00
|
|
|
console_scroll(s, -10);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
|
|
|
case QEMU_KEY_CTRL_PAGEDOWN:
|
2013-03-14 21:27:08 +08:00
|
|
|
console_scroll(s, 10);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
|
|
|
default:
|
2006-06-26 00:26:29 +08: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 20:42:52 +08:00
|
|
|
} else if (s->echo && (keysym == '\r' || keysym == '\n')) {
|
2023-08-30 17:37:38 +08:00
|
|
|
qemu_chr_write(s->chr, (uint8_t *)"\r", 1, true);
|
2010-12-23 20:42:52 +08:00
|
|
|
*q++ = '\n';
|
2006-06-26 00:26:29 +08:00
|
|
|
} else {
|
2010-12-23 20:42:52 +08:00
|
|
|
*q++ = keysym;
|
|
|
|
}
|
|
|
|
if (s->echo) {
|
2023-08-30 17:37:38 +08:00
|
|
|
qemu_chr_write(s->chr, buf, q - buf, true);
|
2006-06-26 00:26:29 +08:00
|
|
|
}
|
2021-09-17 03:22:38 +08:00
|
|
|
num_free = fifo8_num_free(&s->out_fifo);
|
|
|
|
fifo8_push_all(&s->out_fifo, buf, MIN(num_free, q - buf));
|
|
|
|
kbd_send_chars(s);
|
2004-07-15 01:28:59 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
qapi: Don't let implicit enum MAX member collide
Now that we guarantee the user doesn't have any enum values
beginning with a single underscore, we can use that for our
own purposes. Renaming ENUM_MAX to ENUM__MAX makes it obvious
that the sentinel is generated.
This patch was mostly generated by applying a temporary patch:
|diff --git a/scripts/qapi.py b/scripts/qapi.py
|index e6d014b..b862ec9 100644
|--- a/scripts/qapi.py
|+++ b/scripts/qapi.py
|@@ -1570,6 +1570,7 @@ const char *const %(c_name)s_lookup[] = {
| max_index = c_enum_const(name, 'MAX', prefix)
| ret += mcgen('''
| [%(max_index)s] = NULL,
|+// %(max_index)s
| };
| ''',
| max_index=max_index)
then running:
$ cat qapi-{types,event}.c tests/test-qapi-types.c |
sed -n 's,^// \(.*\)MAX,s|\1MAX|\1_MAX|g,p' > list
$ git grep -l _MAX | xargs sed -i -f list
The only things not generated are the changes in scripts/qapi.py.
Rejecting enum members named 'MAX' is now useless, and will be dropped
in the next patch.
Signed-off-by: Eric Blake <eblake@redhat.com>
Message-Id: <1447836791-369-23-git-send-email-eblake@redhat.com>
Reviewed-by: Juan Quintela <quintela@redhat.com>
[Rebased to current master, commit message tweaked]
Signed-off-by: Markus Armbruster <armbru@redhat.com>
2015-11-18 16:52:57 +08:00
|
|
|
static const int qcode_to_keysym[Q_KEY_CODE__MAX] = {
|
2014-05-27 15:28:38 +08:00
|
|
|
[Q_KEY_CODE_UP] = QEMU_KEY_UP,
|
|
|
|
[Q_KEY_CODE_DOWN] = QEMU_KEY_DOWN,
|
|
|
|
[Q_KEY_CODE_RIGHT] = QEMU_KEY_RIGHT,
|
|
|
|
[Q_KEY_CODE_LEFT] = QEMU_KEY_LEFT,
|
|
|
|
[Q_KEY_CODE_HOME] = QEMU_KEY_HOME,
|
|
|
|
[Q_KEY_CODE_END] = QEMU_KEY_END,
|
|
|
|
[Q_KEY_CODE_PGUP] = QEMU_KEY_PAGEUP,
|
|
|
|
[Q_KEY_CODE_PGDN] = QEMU_KEY_PAGEDOWN,
|
|
|
|
[Q_KEY_CODE_DELETE] = QEMU_KEY_DELETE,
|
2022-08-12 06:01:38 +08:00
|
|
|
[Q_KEY_CODE_TAB] = QEMU_KEY_TAB,
|
2016-08-11 15:21:00 +08:00
|
|
|
[Q_KEY_CODE_BACKSPACE] = QEMU_KEY_BACKSPACE,
|
2014-05-27 15:28:38 +08:00
|
|
|
};
|
|
|
|
|
2018-03-21 21:50:36 +08:00
|
|
|
static const int ctrl_qcode_to_keysym[Q_KEY_CODE__MAX] = {
|
|
|
|
[Q_KEY_CODE_UP] = QEMU_KEY_CTRL_UP,
|
|
|
|
[Q_KEY_CODE_DOWN] = QEMU_KEY_CTRL_DOWN,
|
|
|
|
[Q_KEY_CODE_RIGHT] = QEMU_KEY_CTRL_RIGHT,
|
|
|
|
[Q_KEY_CODE_LEFT] = QEMU_KEY_CTRL_LEFT,
|
|
|
|
[Q_KEY_CODE_HOME] = QEMU_KEY_CTRL_HOME,
|
|
|
|
[Q_KEY_CODE_END] = QEMU_KEY_CTRL_END,
|
|
|
|
[Q_KEY_CODE_PGUP] = QEMU_KEY_CTRL_PAGEUP,
|
|
|
|
[Q_KEY_CODE_PGDN] = QEMU_KEY_CTRL_PAGEDOWN,
|
|
|
|
};
|
|
|
|
|
|
|
|
bool kbd_put_qcode_console(QemuConsole *s, int qcode, bool ctrl)
|
2014-05-27 15:28:38 +08:00
|
|
|
{
|
|
|
|
int keysym;
|
|
|
|
|
2018-03-21 21:50:36 +08:00
|
|
|
keysym = ctrl ? ctrl_qcode_to_keysym[qcode] : qcode_to_keysym[qcode];
|
2014-05-27 15:28:38 +08:00
|
|
|
if (keysym == 0) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
kbd_put_keysym_console(s, keysym);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-05-27 15:32:36 +08:00
|
|
|
void kbd_put_string_console(QemuConsole *s, const char *str, int len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < len && str[i]; i++) {
|
|
|
|
kbd_put_keysym_console(s, str[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-22 18:05:52 +08:00
|
|
|
void kbd_put_keysym(int keysym)
|
|
|
|
{
|
|
|
|
kbd_put_keysym_console(active_console, keysym);
|
|
|
|
}
|
|
|
|
|
2008-02-11 00:33:14 +08:00
|
|
|
static void text_console_invalidate(void *opaque)
|
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = QEMU_TEXT_CONSOLE(opaque);
|
2013-03-06 20:40:47 +08:00
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
if (!QEMU_IS_FIXED_TEXT_CONSOLE(s)) {
|
|
|
|
text_console_resize(QEMU_TEXT_CONSOLE(s));
|
2009-01-22 02:59:12 +08:00
|
|
|
}
|
2008-02-11 00:33:14 +08:00
|
|
|
console_refresh(s);
|
|
|
|
}
|
|
|
|
|
2009-10-02 05:12:16 +08:00
|
|
|
static void text_console_update(void *opaque, console_ch_t *chardata)
|
2008-02-11 00:33:14 +08:00
|
|
|
{
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s = QEMU_TEXT_CONSOLE(opaque);
|
2008-02-11 00:33:14 +08: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 ++)
|
2015-11-29 21:28:24 +08:00
|
|
|
for (j = 0; j < s->width; j++, src++) {
|
|
|
|
console_write_ch(chardata ++,
|
|
|
|
ATTR2CHTYPE(s->cells[src].ch,
|
|
|
|
s->cells[src].t_attrib.fgcol,
|
|
|
|
s->cells[src].t_attrib.bgcol,
|
|
|
|
s->cells[src].t_attrib.bold));
|
|
|
|
}
|
2023-08-30 17:38:02 +08:00
|
|
|
dpy_text_update(QEMU_CONSOLE(s), s->text_x[0], s->text_y[0],
|
2012-09-28 21:02:08 +08:00
|
|
|
s->text_x[1] - s->text_x[0], i - s->text_y[0]);
|
2008-02-11 00:33:14 +08: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) {
|
2023-08-30 17:38:02 +08:00
|
|
|
dpy_text_cursor(QEMU_CONSOLE(s), s->x, s->y);
|
2008-02-11 00:33:14 +08:00
|
|
|
s->cursor_invalidate = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:52 +08:00
|
|
|
static void
|
2023-08-30 17:37:54 +08:00
|
|
|
qemu_console_register(QemuConsole *c)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2006-04-09 09:06:34 +08:00
|
|
|
int i;
|
2004-07-15 01:28:59 +08:00
|
|
|
|
2023-08-30 17:37:54 +08:00
|
|
|
if (!active_console || (!QEMU_IS_GRAPHIC_CONSOLE(active_console) &&
|
|
|
|
QEMU_IS_GRAPHIC_CONSOLE(c))) {
|
2023-08-30 17:37:52 +08:00
|
|
|
active_console = c;
|
2007-07-12 07:14:59 +08:00
|
|
|
}
|
2014-05-26 16:36:35 +08:00
|
|
|
|
2018-05-07 17:54:24 +08:00
|
|
|
if (QTAILQ_EMPTY(&consoles)) {
|
2023-08-30 17:37:52 +08:00
|
|
|
c->index = 0;
|
|
|
|
QTAILQ_INSERT_TAIL(&consoles, c, next);
|
2023-08-30 17:37:54 +08:00
|
|
|
} else if (!QEMU_IS_GRAPHIC_CONSOLE(c) || phase_check(PHASE_MACHINE_READY)) {
|
2018-12-06 20:10:34 +08:00
|
|
|
QemuConsole *last = QTAILQ_LAST(&consoles);
|
2023-08-30 17:37:52 +08:00
|
|
|
c->index = last->index + 1;
|
|
|
|
QTAILQ_INSERT_TAIL(&consoles, c, next);
|
2006-04-09 09:06:34 +08:00
|
|
|
} else {
|
2018-03-14 01:17:29 +08:00
|
|
|
/*
|
|
|
|
* HACK: Put graphical consoles before text consoles.
|
|
|
|
*
|
|
|
|
* Only do that for coldplugged devices. After initial device
|
|
|
|
* initialization we will not renumber the consoles any more.
|
|
|
|
*/
|
2023-08-30 17:37:52 +08:00
|
|
|
QemuConsole *it = QTAILQ_FIRST(&consoles);
|
2018-05-07 17:54:24 +08:00
|
|
|
|
2023-08-30 17:37:54 +08:00
|
|
|
while (QTAILQ_NEXT(it, next) != NULL && QEMU_IS_GRAPHIC_CONSOLE(it)) {
|
2023-08-30 17:37:52 +08:00
|
|
|
it = QTAILQ_NEXT(it, next);
|
2018-05-07 17:54:24 +08:00
|
|
|
}
|
2023-08-30 17:37:54 +08:00
|
|
|
if (QEMU_IS_GRAPHIC_CONSOLE(it)) {
|
2018-05-07 17:54:24 +08:00
|
|
|
/* have no text consoles */
|
2023-08-30 17:37:52 +08:00
|
|
|
c->index = it->index + 1;
|
|
|
|
QTAILQ_INSERT_AFTER(&consoles, it, c, next);
|
2018-05-07 17:54:24 +08:00
|
|
|
} else {
|
2023-08-30 17:37:52 +08:00
|
|
|
c->index = it->index;
|
|
|
|
QTAILQ_INSERT_BEFORE(it, c, next);
|
2018-05-07 17:54:24 +08:00
|
|
|
/* renumber text consoles */
|
2023-08-30 17:37:52 +08:00
|
|
|
for (i = c->index + 1; it != NULL; it = QTAILQ_NEXT(it, next), i++) {
|
|
|
|
it->index = i;
|
2018-05-07 17:54:24 +08:00
|
|
|
}
|
2006-04-09 09:06:34 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:51 +08:00
|
|
|
static void
|
|
|
|
qemu_console_finalize(Object *obj)
|
|
|
|
{
|
2023-08-30 17:37:59 +08:00
|
|
|
QemuConsole *c = QEMU_CONSOLE(obj);
|
|
|
|
|
2023-08-30 17:38:01 +08:00
|
|
|
/* TODO: check this code path, and unregister from consoles */
|
|
|
|
g_clear_pointer(&c->surface, qemu_free_displaysurface);
|
|
|
|
g_clear_pointer(&c->gl_unblock_timer, timer_free);
|
2023-08-30 17:37:59 +08:00
|
|
|
g_clear_pointer(&c->ui_timer, timer_free);
|
2023-08-30 17:37:52 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:51 +08:00
|
|
|
static void
|
|
|
|
qemu_console_class_init(ObjectClass *oc, void *data)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_console_init(Object *obj)
|
|
|
|
{
|
2023-08-30 17:37:52 +08:00
|
|
|
QemuConsole *c = QEMU_CONSOLE(obj);
|
|
|
|
DisplayState *ds = get_alloc_displaystate();
|
|
|
|
|
|
|
|
qemu_co_queue_init(&c->dump_queue);
|
|
|
|
c->ds = ds;
|
|
|
|
c->window_id = -1;
|
2023-08-30 17:37:59 +08:00
|
|
|
c->ui_timer = timer_new_ms(QEMU_CLOCK_REALTIME,
|
|
|
|
dpy_set_ui_info_timer, c);
|
2023-08-30 17:37:55 +08:00
|
|
|
qemu_console_register(c);
|
2023-08-30 17:37:52 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:53 +08:00
|
|
|
static void
|
|
|
|
qemu_graphic_console_finalize(Object *obj)
|
|
|
|
{
|
2023-08-30 17:38:03 +08:00
|
|
|
QemuGraphicConsole *c = QEMU_GRAPHIC_CONSOLE(obj);
|
|
|
|
|
|
|
|
g_clear_pointer(&c->device, object_unref);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_graphic_console_prop_get_head(Object *obj, Visitor *v, const char *name,
|
|
|
|
void *opaque, Error **errp)
|
|
|
|
{
|
|
|
|
QemuGraphicConsole *c = QEMU_GRAPHIC_CONSOLE(obj);
|
|
|
|
|
|
|
|
visit_type_uint32(v, name, &c->head, errp);
|
2023-08-30 17:37:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_graphic_console_class_init(ObjectClass *oc, void *data)
|
|
|
|
{
|
2023-08-30 17:38:03 +08:00
|
|
|
object_class_property_add_link(oc, "device", TYPE_DEVICE,
|
|
|
|
offsetof(QemuGraphicConsole, device),
|
|
|
|
object_property_allow_set_link,
|
|
|
|
OBJ_PROP_LINK_STRONG);
|
|
|
|
object_class_property_add(oc, "head", "uint32",
|
|
|
|
qemu_graphic_console_prop_get_head,
|
|
|
|
NULL, NULL, NULL);
|
2023-08-30 17:37:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_graphic_console_init(Object *obj)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_text_console_finalize(Object *obj)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_text_console_class_init(ObjectClass *oc, void *data)
|
|
|
|
{
|
2023-08-30 17:38:00 +08:00
|
|
|
if (!cursor_timer) {
|
|
|
|
cursor_timer = timer_new_ms(QEMU_CLOCK_REALTIME,
|
|
|
|
text_console_update_cursor, NULL);
|
|
|
|
}
|
2023-08-30 17:37:53 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:05 +08:00
|
|
|
static const GraphicHwOps text_console_ops = {
|
|
|
|
.invalidate = text_console_invalidate,
|
|
|
|
.text_update = text_console_update,
|
|
|
|
};
|
|
|
|
|
2023-08-30 17:37:53 +08:00
|
|
|
static void
|
|
|
|
qemu_text_console_init(Object *obj)
|
|
|
|
{
|
2023-08-30 17:38:05 +08:00
|
|
|
QemuTextConsole *c = QEMU_TEXT_CONSOLE(obj);
|
|
|
|
|
|
|
|
fifo8_create(&c->out_fifo, 16);
|
|
|
|
c->total_height = DEFAULT_BACKSCROLL;
|
|
|
|
QEMU_CONSOLE(c)->hw_ops = &text_console_ops;
|
|
|
|
QEMU_CONSOLE(c)->hw = c;
|
2023-08-30 17:37:53 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_fixed_text_console_finalize(Object *obj)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_fixed_text_console_class_init(ObjectClass *oc, void *data)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
qemu_fixed_text_console_init(Object *obj)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2023-06-06 19:56:46 +08:00
|
|
|
#ifdef WIN32
|
|
|
|
void qemu_displaysurface_win32_set_handle(DisplaySurface *surface,
|
|
|
|
HANDLE h, uint32_t offset)
|
|
|
|
{
|
|
|
|
assert(!surface->handle);
|
|
|
|
|
|
|
|
surface->handle = h;
|
|
|
|
surface->handle_offset = offset;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
win32_pixman_image_destroy(pixman_image_t *image, void *data)
|
|
|
|
{
|
|
|
|
DisplaySurface *surface = data;
|
|
|
|
|
|
|
|
if (!surface->handle) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(surface->handle_offset == 0);
|
|
|
|
|
|
|
|
qemu_win32_map_free(
|
|
|
|
pixman_image_get_data(surface->image),
|
|
|
|
surface->handle,
|
|
|
|
&error_warn
|
|
|
|
);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-03-12 18:00:42 +08:00
|
|
|
DisplaySurface *qemu_create_displaysurface(int width, int height)
|
2011-03-16 20:33:30 +08:00
|
|
|
{
|
2023-06-06 19:56:46 +08:00
|
|
|
DisplaySurface *surface;
|
|
|
|
void *bits = NULL;
|
|
|
|
#ifdef WIN32
|
|
|
|
HANDLE handle = NULL;
|
|
|
|
#endif
|
2012-09-26 21:20:05 +08:00
|
|
|
|
2023-06-06 19:56:46 +08:00
|
|
|
trace_displaysurface_create(width, height);
|
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
bits = qemu_win32_map_alloc(width * height * 4, &handle, &error_abort);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
surface = qemu_create_displaysurface_from(
|
|
|
|
width, height,
|
|
|
|
PIXMAN_x8r8g8b8,
|
|
|
|
width * 4, bits
|
|
|
|
);
|
2014-06-18 17:03:15 +08:00
|
|
|
surface->flags = QEMU_ALLOCATED_FLAG;
|
2010-02-11 07:29:57 +08:00
|
|
|
|
2023-06-06 19:56:46 +08:00
|
|
|
#ifdef WIN32
|
|
|
|
qemu_displaysurface_win32_set_handle(surface, handle, 0);
|
|
|
|
#endif
|
2012-09-27 17:06:36 +08:00
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
2014-06-18 17:03:15 +08:00
|
|
|
DisplaySurface *qemu_create_displaysurface_from(int width, int height,
|
|
|
|
pixman_format_code_t format,
|
|
|
|
int linesize, uint8_t *data)
|
2010-02-11 07:29:57 +08:00
|
|
|
{
|
2012-09-26 21:20:05 +08:00
|
|
|
DisplaySurface *surface = g_new0(DisplaySurface, 1);
|
2010-02-11 07:29:57 +08:00
|
|
|
|
2014-06-18 17:03:15 +08:00
|
|
|
trace_displaysurface_create_from(surface, width, height, format);
|
|
|
|
surface->format = format;
|
2012-09-26 21:20:05 +08:00
|
|
|
surface->image = pixman_image_create_bits(surface->format,
|
|
|
|
width, height,
|
|
|
|
(void *)data, linesize);
|
|
|
|
assert(surface->image != NULL);
|
2023-06-06 19:56:46 +08:00
|
|
|
#ifdef WIN32
|
|
|
|
pixman_image_set_destroy_function(surface->image,
|
|
|
|
win32_pixman_image_destroy, surface);
|
|
|
|
#endif
|
2012-09-26 21:20:05 +08:00
|
|
|
|
2010-02-11 07:29:57 +08:00
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
2016-04-01 16:27:20 +08:00
|
|
|
DisplaySurface *qemu_create_displaysurface_pixman(pixman_image_t *image)
|
|
|
|
{
|
|
|
|
DisplaySurface *surface = g_new0(DisplaySurface, 1);
|
|
|
|
|
|
|
|
trace_displaysurface_create_pixman(surface);
|
|
|
|
surface->format = pixman_image_get_format(image);
|
|
|
|
surface->image = pixman_image_ref(image);
|
|
|
|
|
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
2021-02-25 18:13:14 +08:00
|
|
|
DisplaySurface *qemu_create_placeholder_surface(int w, int h,
|
|
|
|
const char *msg)
|
2013-04-25 15:33:19 +08:00
|
|
|
{
|
2013-04-25 18:10:45 +08:00
|
|
|
DisplaySurface *surface = qemu_create_displaysurface(w, h);
|
2023-08-30 17:38:10 +08:00
|
|
|
pixman_color_t bg = QEMU_PIXMAN_COLOR_BLACK;
|
|
|
|
pixman_color_t fg = QEMU_PIXMAN_COLOR_GRAY;
|
2013-04-25 15:33:19 +08:00
|
|
|
pixman_image_t *glyph;
|
|
|
|
int len, x, y, i;
|
|
|
|
|
|
|
|
len = strlen(msg);
|
2013-04-25 18:10:45 +08:00
|
|
|
x = (w / FONT_WIDTH - len) / 2;
|
|
|
|
y = (h / FONT_HEIGHT - 1) / 2;
|
2013-04-25 15:33:19 +08:00
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
glyph = qemu_pixman_glyph_from_vgafont(FONT_HEIGHT, vgafont16, msg[i]);
|
|
|
|
qemu_pixman_glyph_render(glyph, surface->image, &fg, &bg,
|
|
|
|
x+i, y, FONT_WIDTH, FONT_HEIGHT);
|
|
|
|
qemu_pixman_image_unref(glyph);
|
|
|
|
}
|
2021-02-25 18:13:14 +08:00
|
|
|
surface->flags |= QEMU_PLACEHOLDER_FLAG;
|
2013-04-25 15:33:19 +08:00
|
|
|
return surface;
|
|
|
|
}
|
|
|
|
|
2013-02-28 17:48:02 +08:00
|
|
|
void qemu_free_displaysurface(DisplaySurface *surface)
|
2010-02-11 07:29:57 +08:00
|
|
|
{
|
2013-02-28 17:48:02 +08:00
|
|
|
if (surface == NULL) {
|
2010-02-11 07:29:57 +08:00
|
|
|
return;
|
2012-09-26 13:46:20 +08:00
|
|
|
}
|
2013-02-28 17:48:02 +08:00
|
|
|
trace_displaysurface_free(surface);
|
|
|
|
qemu_pixman_image_unref(surface->image);
|
|
|
|
g_free(surface);
|
2010-02-11 07:29:57 +08:00
|
|
|
}
|
|
|
|
|
2014-07-11 19:56:51 +08:00
|
|
|
bool console_has_gl(QemuConsole *con)
|
|
|
|
{
|
|
|
|
return con->gl != NULL;
|
|
|
|
}
|
|
|
|
|
2021-02-04 18:52:24 +08:00
|
|
|
static bool displaychangelistener_has_dmabuf(DisplayChangeListener *dcl)
|
|
|
|
{
|
|
|
|
if (dcl->ops->dpy_has_dmabuf) {
|
|
|
|
return dcl->ops->dpy_has_dmabuf(dcl);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dcl->ops->dpy_gl_scanout_dmabuf) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-02-17 00:16:55 +08:00
|
|
|
static bool console_compatible_with(QemuConsole *con,
|
|
|
|
DisplayChangeListener *dcl, Error **errp)
|
2021-02-04 18:52:25 +08:00
|
|
|
{
|
|
|
|
int flags;
|
|
|
|
|
|
|
|
flags = con->hw_ops->get_flags ? con->hw_ops->get_flags(con->hw) : 0;
|
|
|
|
|
2022-02-16 23:33:37 +08:00
|
|
|
if (console_has_gl(con) &&
|
|
|
|
!con->gl->ops->dpy_gl_ctx_is_compatible_dcl(con->gl, dcl)) {
|
2022-02-16 21:35:28 +08:00
|
|
|
error_setg(errp, "Display %s is incompatible with the GL context",
|
|
|
|
dcl->ops->dpy_name);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-02-04 18:52:25 +08:00
|
|
|
if (flags & GRAPHIC_FLAGS_GL &&
|
|
|
|
!console_has_gl(con)) {
|
|
|
|
error_setg(errp, "The console requires a GL context.");
|
|
|
|
return false;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flags & GRAPHIC_FLAGS_DMABUF &&
|
|
|
|
!displaychangelistener_has_dmabuf(dcl)) {
|
|
|
|
error_setg(errp, "The console requires display DMABUF support.");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-06-19 17:53:36 +08:00
|
|
|
void console_handle_touch_event(QemuConsole *con,
|
|
|
|
struct touch_slot touch_slots[INPUT_EVENT_SLOTS_MAX],
|
|
|
|
uint64_t num_slot,
|
|
|
|
int width, int height,
|
|
|
|
double x, double y,
|
|
|
|
InputMultiTouchType type,
|
|
|
|
Error **errp)
|
|
|
|
{
|
|
|
|
struct touch_slot *slot;
|
|
|
|
bool needs_sync = false;
|
|
|
|
int update;
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (num_slot >= INPUT_EVENT_SLOTS_MAX) {
|
|
|
|
error_setg(errp,
|
|
|
|
"Unexpected touch slot number: % " PRId64" >= %d",
|
|
|
|
num_slot, INPUT_EVENT_SLOTS_MAX);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
slot = &touch_slots[num_slot];
|
|
|
|
slot->x = x;
|
|
|
|
slot->y = y;
|
|
|
|
|
|
|
|
if (type == INPUT_MULTI_TOUCH_TYPE_BEGIN) {
|
|
|
|
slot->tracking_id = num_slot;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < INPUT_EVENT_SLOTS_MAX; ++i) {
|
|
|
|
if (i == num_slot) {
|
|
|
|
update = type;
|
|
|
|
} else {
|
|
|
|
update = INPUT_MULTI_TOUCH_TYPE_UPDATE;
|
|
|
|
}
|
|
|
|
|
|
|
|
slot = &touch_slots[i];
|
|
|
|
|
|
|
|
if (slot->tracking_id == -1) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (update == INPUT_MULTI_TOUCH_TYPE_END) {
|
|
|
|
slot->tracking_id = -1;
|
|
|
|
qemu_input_queue_mtt(con, update, i, slot->tracking_id);
|
|
|
|
needs_sync = true;
|
|
|
|
} else {
|
|
|
|
qemu_input_queue_mtt(con, update, i, slot->tracking_id);
|
|
|
|
qemu_input_queue_btn(con, INPUT_BUTTON_TOUCH, true);
|
|
|
|
qemu_input_queue_mtt_abs(con,
|
|
|
|
INPUT_AXIS_X, (int) slot->x,
|
|
|
|
0, width,
|
|
|
|
i, slot->tracking_id);
|
|
|
|
qemu_input_queue_mtt_abs(con,
|
|
|
|
INPUT_AXIS_Y, (int) slot->y,
|
|
|
|
0, height,
|
|
|
|
i, slot->tracking_id);
|
|
|
|
needs_sync = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (needs_sync) {
|
|
|
|
qemu_input_event_sync();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-10 03:48:46 +08:00
|
|
|
void qemu_console_set_display_gl_ctx(QemuConsole *con, DisplayGLCtx *gl)
|
2021-01-25 18:53:18 +08:00
|
|
|
{
|
|
|
|
/* display has opengl support */
|
2021-10-10 03:48:46 +08:00
|
|
|
assert(con);
|
|
|
|
if (con->gl) {
|
|
|
|
error_report("The console already has an OpenGL context.");
|
2021-01-25 18:53:18 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
2021-10-10 03:48:46 +08:00
|
|
|
con->gl = gl;
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:03 +08:00
|
|
|
static void
|
|
|
|
dcl_set_graphic_cursor(DisplayChangeListener *dcl, QemuGraphicConsole *con)
|
|
|
|
{
|
|
|
|
if (con && con->cursor && dcl->ops->dpy_cursor_define) {
|
|
|
|
dcl->ops->dpy_cursor_define(dcl, con->cursor);
|
|
|
|
}
|
|
|
|
if (con && dcl->ops->dpy_mouse_set) {
|
|
|
|
dcl->ops->dpy_mouse_set(dcl, con->cursor_x, con->cursor_y, con->cursor_on);
|
|
|
|
}
|
|
|
|
}
|
2013-04-23 21:44:31 +08:00
|
|
|
void register_displaychangelistener(DisplayChangeListener *dcl)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2013-03-15 22:45:54 +08:00
|
|
|
QemuConsole *con;
|
|
|
|
|
2017-04-06 20:05:12 +08:00
|
|
|
assert(!dcl->ds);
|
|
|
|
|
2012-11-13 21:51:41 +08:00
|
|
|
trace_displaychangelistener_register(dcl, dcl->ops->dpy_name);
|
2013-04-23 21:44:31 +08:00
|
|
|
dcl->ds = get_alloc_displaystate();
|
|
|
|
QLIST_INSERT_HEAD(&dcl->ds->listeners, dcl, next);
|
|
|
|
gui_setup_refresh(dcl->ds);
|
2013-03-15 22:45:54 +08:00
|
|
|
if (dcl->con) {
|
|
|
|
dcl->con->dcls++;
|
|
|
|
con = dcl->con;
|
|
|
|
} else {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2022-02-17 00:16:55 +08:00
|
|
|
displaychangelistener_display_console(dcl, con, dcl->con ? &error_fatal : NULL);
|
2023-08-30 17:38:03 +08:00
|
|
|
if (QEMU_IS_GRAPHIC_CONSOLE(con)) {
|
|
|
|
dcl_set_graphic_cursor(dcl, QEMU_GRAPHIC_CONSOLE(con));
|
2023-01-17 19:40:58 +08:00
|
|
|
}
|
2014-05-22 17:27:13 +08:00
|
|
|
text_console_update_cursor(NULL);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
|
2013-03-14 18:56:16 +08:00
|
|
|
void update_displaychangelistener(DisplayChangeListener *dcl,
|
|
|
|
uint64_t interval)
|
|
|
|
{
|
|
|
|
DisplayState *ds = dcl->ds;
|
|
|
|
|
|
|
|
dcl->update_interval = interval;
|
|
|
|
if (!ds->refreshing && ds->update_interval > interval) {
|
2013-08-21 23:03:08 +08:00
|
|
|
timer_mod(ds->gui_timer, ds->last_update + interval);
|
2013-03-14 18:56:16 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-11-13 21:51:41 +08:00
|
|
|
void unregister_displaychangelistener(DisplayChangeListener *dcl)
|
|
|
|
{
|
|
|
|
DisplayState *ds = dcl->ds;
|
|
|
|
trace_displaychangelistener_unregister(dcl, dcl->ops->dpy_name);
|
2013-03-15 22:45:54 +08:00
|
|
|
if (dcl->con) {
|
|
|
|
dcl->con->dcls--;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_REMOVE(dcl, next);
|
2017-11-09 18:51:54 +08:00
|
|
|
dcl->ds = NULL;
|
2012-11-13 21:51:41 +08:00
|
|
|
gui_setup_refresh(ds);
|
|
|
|
}
|
|
|
|
|
2015-03-12 19:51:13 +08:00
|
|
|
static void dpy_set_ui_info_timer(void *opaque)
|
|
|
|
{
|
|
|
|
QemuConsole *con = opaque;
|
2023-08-30 17:38:03 +08:00
|
|
|
uint32_t head = qemu_console_get_head(con);
|
2015-03-12 19:51:13 +08:00
|
|
|
|
2023-08-30 17:38:03 +08:00
|
|
|
con->hw_ops->ui_info(con->hw, head, &con->ui_info);
|
2015-03-12 19:51:13 +08:00
|
|
|
}
|
|
|
|
|
2015-03-13 19:21:14 +08:00
|
|
|
bool dpy_ui_info_supported(QemuConsole *con)
|
|
|
|
{
|
2020-12-08 19:57:30 +08:00
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
|
|
|
|
2015-03-13 19:21:14 +08:00
|
|
|
return con->hw_ops->ui_info != NULL;
|
|
|
|
}
|
|
|
|
|
2020-09-27 22:57:48 +08:00
|
|
|
const QemuUIInfo *dpy_get_ui_info(const QemuConsole *con)
|
|
|
|
{
|
2020-12-08 19:57:30 +08:00
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2020-09-27 22:57:48 +08:00
|
|
|
|
|
|
|
return &con->ui_info;
|
|
|
|
}
|
|
|
|
|
2021-04-14 00:39:11 +08:00
|
|
|
int dpy_set_ui_info(QemuConsole *con, QemuUIInfo *info, bool delay)
|
2014-01-25 00:38:20 +08:00
|
|
|
{
|
2020-12-08 19:57:30 +08:00
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2016-05-30 16:41:13 +08:00
|
|
|
|
2015-03-13 19:21:14 +08:00
|
|
|
if (!dpy_ui_info_supported(con)) {
|
2015-03-12 19:51:13 +08:00
|
|
|
return -1;
|
2014-01-25 00:38:20 +08:00
|
|
|
}
|
2016-05-30 16:41:13 +08:00
|
|
|
if (memcmp(&con->ui_info, info, sizeof(con->ui_info)) == 0) {
|
|
|
|
/* nothing changed -- ignore */
|
|
|
|
return 0;
|
|
|
|
}
|
2015-03-12 19:51:13 +08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Typically we get a flood of these as the user resizes the window.
|
|
|
|
* Wait until the dust has settled (one second without updates), then
|
|
|
|
* go notify the guest.
|
|
|
|
*/
|
2016-05-30 16:41:13 +08:00
|
|
|
con->ui_info = *info;
|
2021-04-14 00:39:11 +08:00
|
|
|
timer_mod(con->ui_timer,
|
|
|
|
qemu_clock_get_ms(QEMU_CLOCK_REALTIME) + (delay ? 1000 : 0));
|
2015-03-12 19:51:13 +08:00
|
|
|
return 0;
|
2014-01-25 00:38:20 +08:00
|
|
|
}
|
|
|
|
|
2013-03-05 22:24:14 +08:00
|
|
|
void dpy_gfx_update(QemuConsole *con, int x, int y, int w, int h)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2013-03-05 22:24:14 +08:00
|
|
|
DisplayState *s = con->ds;
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2021-02-20 20:23:03 +08:00
|
|
|
int width = qemu_console_get_width(con, x + w);
|
|
|
|
int height = qemu_console_get_height(con, y + h);
|
2012-11-13 21:51:41 +08:00
|
|
|
|
|
|
|
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);
|
|
|
|
|
2013-03-14 21:27:08 +08:00
|
|
|
if (!qemu_console_is_visible(con)) {
|
2013-03-12 21:39:22 +08:00
|
|
|
return;
|
|
|
|
}
|
2022-02-17 19:07:21 +08:00
|
|
|
dpy_gfx_update_texture(con, con->surface, x, y, w, h);
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2013-03-15 22:45:54 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
if (dcl->ops->dpy_gfx_update) {
|
2013-03-01 20:03:04 +08:00
|
|
|
dcl->ops->dpy_gfx_update(dcl, x, y, w, h);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-27 17:11:05 +08:00
|
|
|
void dpy_gfx_update_full(QemuConsole *con)
|
|
|
|
{
|
2021-02-20 20:23:03 +08:00
|
|
|
int w = qemu_console_get_width(con, 0);
|
|
|
|
int h = qemu_console_get_height(con, 0);
|
|
|
|
|
|
|
|
dpy_gfx_update(con, 0, 0, w, h);
|
2018-04-27 17:11:05 +08:00
|
|
|
}
|
|
|
|
|
2013-03-12 21:39:22 +08:00
|
|
|
void dpy_gfx_replace_surface(QemuConsole *con,
|
|
|
|
DisplaySurface *surface)
|
|
|
|
{
|
2021-02-25 18:13:15 +08:00
|
|
|
static const char placeholder_msg[] = "Display output is not active.";
|
2013-03-12 21:39:22 +08:00
|
|
|
DisplayState *s = con->ds;
|
|
|
|
DisplaySurface *old_surface = con->surface;
|
2023-06-28 06:44:51 +08:00
|
|
|
DisplaySurface *new_surface = surface;
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2021-02-25 18:13:15 +08:00
|
|
|
int width;
|
|
|
|
int height;
|
|
|
|
|
|
|
|
if (!surface) {
|
|
|
|
if (old_surface) {
|
|
|
|
width = surface_width(old_surface);
|
|
|
|
height = surface_height(old_surface);
|
|
|
|
} else {
|
|
|
|
width = 640;
|
|
|
|
height = 480;
|
|
|
|
}
|
|
|
|
|
2023-06-28 06:44:51 +08:00
|
|
|
new_surface = qemu_create_placeholder_surface(width, height, placeholder_msg);
|
2021-02-25 18:13:15 +08:00
|
|
|
}
|
2013-03-12 21:39:22 +08:00
|
|
|
|
2023-06-28 06:44:51 +08:00
|
|
|
assert(old_surface != new_surface);
|
2017-04-06 20:05:11 +08:00
|
|
|
|
2021-02-20 20:23:03 +08:00
|
|
|
con->scanout.kind = SCANOUT_SURFACE;
|
2023-06-28 06:44:51 +08:00
|
|
|
con->surface = new_surface;
|
|
|
|
dpy_gfx_create_texture(con, new_surface);
|
2013-03-15 22:45:54 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2023-06-28 06:44:51 +08:00
|
|
|
displaychangelistener_gfx_switch(dcl, new_surface, surface ? FALSE : TRUE);
|
2013-03-12 21:39:22 +08:00
|
|
|
}
|
2022-02-17 19:07:21 +08:00
|
|
|
dpy_gfx_destroy_texture(con, old_surface);
|
2013-02-28 17:48:02 +08:00
|
|
|
qemu_free_displaysurface(old_surface);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
|
2014-07-07 14:39:05 +08:00
|
|
|
bool dpy_gfx_check_format(QemuConsole *con,
|
|
|
|
pixman_format_code_t format)
|
|
|
|
{
|
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
DisplayState *s = con->ds;
|
|
|
|
|
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->con && dcl->con != con) {
|
|
|
|
/* dcl bound to another console -> skip */
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (dcl->ops->dpy_gfx_check_format) {
|
|
|
|
if (!dcl->ops->dpy_gfx_check_format(dcl, format)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
2021-03-04 02:46:40 +08:00
|
|
|
/* default is to allow native 32 bpp only */
|
2014-07-07 14:39:05 +08:00
|
|
|
if (format != qemu_default_pixman_format(32, true)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-05-03 04:48:30 +08:00
|
|
|
static void dpy_refresh(DisplayState *s)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_refresh) {
|
2017-06-14 16:45:38 +08:00
|
|
|
dcl->ops->dpy_refresh(dcl);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 22:24:14 +08:00
|
|
|
void dpy_text_cursor(QemuConsole *con, int x, int y)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2013-03-05 22:24:14 +08:00
|
|
|
DisplayState *s = con->ds;
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2013-03-12 21:39:22 +08:00
|
|
|
|
2013-03-14 21:27:08 +08:00
|
|
|
if (!qemu_console_is_visible(con)) {
|
2013-03-12 21:39:22 +08:00
|
|
|
return;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2013-03-15 22:45:54 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
if (dcl->ops->dpy_text_cursor) {
|
2013-03-01 20:03:04 +08:00
|
|
|
dcl->ops->dpy_text_cursor(dcl, x, y);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 22:24:14 +08:00
|
|
|
void dpy_text_update(QemuConsole *con, int x, int y, int w, int h)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2013-03-05 22:24:14 +08:00
|
|
|
DisplayState *s = con->ds;
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2013-03-12 21:39:22 +08:00
|
|
|
|
2013-03-14 21:27:08 +08:00
|
|
|
if (!qemu_console_is_visible(con)) {
|
2013-03-12 21:39:22 +08:00
|
|
|
return;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2013-03-15 22:45:54 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
if (dcl->ops->dpy_text_update) {
|
2013-03-01 20:03:04 +08:00
|
|
|
dcl->ops->dpy_text_update(dcl, x, y, w, h);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 22:24:14 +08:00
|
|
|
void dpy_text_resize(QemuConsole *con, int w, int h)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2013-03-05 22:24:14 +08:00
|
|
|
DisplayState *s = con->ds;
|
2015-04-09 02:04:13 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2013-03-12 21:39:22 +08:00
|
|
|
|
2013-03-14 21:27:08 +08:00
|
|
|
if (!qemu_console_is_visible(con)) {
|
2013-03-12 21:39:22 +08:00
|
|
|
return;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2013-03-15 22:45:54 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
if (dcl->ops->dpy_text_resize) {
|
2013-03-01 20:03:04 +08:00
|
|
|
dcl->ops->dpy_text_resize(dcl, w, h);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:03 +08:00
|
|
|
void dpy_mouse_set(QemuConsole *c, int x, int y, int on)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2023-08-30 17:38:03 +08:00
|
|
|
QemuGraphicConsole *con = QEMU_GRAPHIC_CONSOLE(c);
|
|
|
|
DisplayState *s = c->ds;
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2013-03-12 21:39:22 +08:00
|
|
|
|
2023-01-17 19:40:58 +08:00
|
|
|
con->cursor_x = x;
|
|
|
|
con->cursor_y = y;
|
|
|
|
con->cursor_on = on;
|
2023-08-30 17:38:03 +08:00
|
|
|
if (!qemu_console_is_visible(c)) {
|
2013-03-12 21:39:22 +08:00
|
|
|
return;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2023-08-30 17:38:03 +08:00
|
|
|
if (c != (dcl->con ? dcl->con : active_console)) {
|
2013-03-15 22:45:54 +08:00
|
|
|
continue;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
if (dcl->ops->dpy_mouse_set) {
|
2013-03-01 20:03:04 +08:00
|
|
|
dcl->ops->dpy_mouse_set(dcl, x, y, on);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:03 +08:00
|
|
|
void dpy_cursor_define(QemuConsole *c, QEMUCursor *cursor)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2023-08-30 17:38:03 +08:00
|
|
|
QemuGraphicConsole *con = QEMU_GRAPHIC_CONSOLE(c);
|
|
|
|
DisplayState *s = c->ds;
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
2013-03-12 21:39:22 +08:00
|
|
|
|
2023-01-17 19:24:40 +08:00
|
|
|
cursor_unref(con->cursor);
|
|
|
|
con->cursor = cursor_ref(cursor);
|
2023-08-30 17:38:03 +08:00
|
|
|
if (!qemu_console_is_visible(c)) {
|
2013-03-12 21:39:22 +08:00
|
|
|
return;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2023-08-30 17:38:03 +08:00
|
|
|
if (c != (dcl->con ? dcl->con : active_console)) {
|
2013-03-15 22:45:54 +08:00
|
|
|
continue;
|
|
|
|
}
|
2012-11-13 21:51:41 +08:00
|
|
|
if (dcl->ops->dpy_cursor_define) {
|
2013-03-01 20:03:04 +08:00
|
|
|
dcl->ops->dpy_cursor_define(dcl, cursor);
|
2012-11-13 21:51:41 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-05 22:24:14 +08:00
|
|
|
bool dpy_cursor_define_supported(QemuConsole *con)
|
2012-11-13 21:51:41 +08:00
|
|
|
{
|
2013-03-05 22:24:14 +08:00
|
|
|
DisplayState *s = con->ds;
|
2013-03-15 22:45:54 +08:00
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
|
2012-11-13 21:51:41 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
|
|
|
if (dcl->ops->dpy_cursor_define) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2014-07-11 19:56:51 +08:00
|
|
|
QEMUGLContext dpy_gl_ctx_create(QemuConsole *con,
|
|
|
|
struct QEMUGLParams *qparams)
|
|
|
|
{
|
|
|
|
assert(con->gl);
|
|
|
|
return con->gl->ops->dpy_gl_ctx_create(con->gl, qparams);
|
|
|
|
}
|
|
|
|
|
|
|
|
void dpy_gl_ctx_destroy(QemuConsole *con, QEMUGLContext ctx)
|
|
|
|
{
|
|
|
|
assert(con->gl);
|
|
|
|
con->gl->ops->dpy_gl_ctx_destroy(con->gl, ctx);
|
|
|
|
}
|
|
|
|
|
|
|
|
int dpy_gl_ctx_make_current(QemuConsole *con, QEMUGLContext ctx)
|
|
|
|
{
|
|
|
|
assert(con->gl);
|
|
|
|
return con->gl->ops->dpy_gl_ctx_make_current(con->gl, ctx);
|
|
|
|
}
|
|
|
|
|
2017-02-21 17:37:17 +08:00
|
|
|
void dpy_gl_scanout_disable(QemuConsole *con)
|
|
|
|
{
|
2021-01-26 04:00:30 +08:00
|
|
|
DisplayState *s = con->ds;
|
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
|
2021-02-20 20:23:03 +08:00
|
|
|
if (con->scanout.kind != SCANOUT_SURFACE) {
|
|
|
|
con->scanout.kind = SCANOUT_NONE;
|
|
|
|
}
|
2021-01-26 04:00:30 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2022-03-26 00:12:16 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-02-15 04:13:35 +08:00
|
|
|
if (dcl->ops->dpy_gl_scanout_disable) {
|
|
|
|
dcl->ops->dpy_gl_scanout_disable(dcl);
|
|
|
|
}
|
2021-01-26 04:00:30 +08:00
|
|
|
}
|
2017-02-21 17:37:17 +08:00
|
|
|
}
|
|
|
|
|
2017-02-21 17:37:16 +08:00
|
|
|
void dpy_gl_scanout_texture(QemuConsole *con,
|
|
|
|
uint32_t backing_id,
|
|
|
|
bool backing_y_0_top,
|
|
|
|
uint32_t backing_width,
|
|
|
|
uint32_t backing_height,
|
|
|
|
uint32_t x, uint32_t y,
|
2023-06-06 19:56:56 +08:00
|
|
|
uint32_t width, uint32_t height,
|
|
|
|
void *d3d_tex2d)
|
2014-07-11 19:56:51 +08:00
|
|
|
{
|
2021-01-26 04:00:30 +08:00
|
|
|
DisplayState *s = con->ds;
|
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
|
2021-02-20 20:23:03 +08:00
|
|
|
con->scanout.kind = SCANOUT_TEXTURE;
|
|
|
|
con->scanout.texture = (ScanoutTexture) {
|
|
|
|
backing_id, backing_y_0_top, backing_width, backing_height,
|
2023-06-06 19:56:56 +08:00
|
|
|
x, y, width, height, d3d_tex2d,
|
2021-02-20 20:23:03 +08:00
|
|
|
};
|
2021-01-26 04:00:30 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2022-03-26 00:12:16 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-02-15 04:13:35 +08:00
|
|
|
if (dcl->ops->dpy_gl_scanout_texture) {
|
|
|
|
dcl->ops->dpy_gl_scanout_texture(dcl, backing_id,
|
|
|
|
backing_y_0_top,
|
|
|
|
backing_width, backing_height,
|
2023-06-06 19:56:56 +08:00
|
|
|
x, y, width, height,
|
|
|
|
d3d_tex2d);
|
2022-02-15 04:13:35 +08:00
|
|
|
}
|
2021-01-26 04:00:30 +08:00
|
|
|
}
|
2014-07-11 19:56:51 +08:00
|
|
|
}
|
|
|
|
|
2017-10-10 21:54:48 +08:00
|
|
|
void dpy_gl_scanout_dmabuf(QemuConsole *con,
|
|
|
|
QemuDmaBuf *dmabuf)
|
|
|
|
{
|
2021-01-26 04:00:30 +08:00
|
|
|
DisplayState *s = con->ds;
|
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
|
2021-02-20 20:23:03 +08:00
|
|
|
con->scanout.kind = SCANOUT_DMABUF;
|
|
|
|
con->scanout.dmabuf = dmabuf;
|
2021-01-26 04:00:30 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2022-03-26 00:12:16 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-02-15 04:13:35 +08:00
|
|
|
if (dcl->ops->dpy_gl_scanout_dmabuf) {
|
|
|
|
dcl->ops->dpy_gl_scanout_dmabuf(dcl, dmabuf);
|
|
|
|
}
|
2021-01-26 04:00:30 +08:00
|
|
|
}
|
2017-10-10 21:54:48 +08:00
|
|
|
}
|
|
|
|
|
2018-02-20 19:04:31 +08:00
|
|
|
void dpy_gl_cursor_dmabuf(QemuConsole *con, QemuDmaBuf *dmabuf,
|
|
|
|
bool have_hot, uint32_t hot_x, uint32_t hot_y)
|
2017-10-10 21:54:48 +08:00
|
|
|
{
|
2021-01-26 04:00:30 +08:00
|
|
|
DisplayState *s = con->ds;
|
|
|
|
DisplayChangeListener *dcl;
|
2017-10-10 21:54:48 +08:00
|
|
|
|
2021-01-26 04:00:30 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2022-03-26 00:12:16 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-01-26 04:00:30 +08:00
|
|
|
if (dcl->ops->dpy_gl_cursor_dmabuf) {
|
|
|
|
dcl->ops->dpy_gl_cursor_dmabuf(dcl, dmabuf,
|
2018-02-20 19:04:31 +08:00
|
|
|
have_hot, hot_x, hot_y);
|
2021-01-26 04:00:30 +08:00
|
|
|
}
|
2018-02-20 19:04:31 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void dpy_gl_cursor_position(QemuConsole *con,
|
|
|
|
uint32_t pos_x, uint32_t pos_y)
|
|
|
|
{
|
2021-01-26 04:00:30 +08:00
|
|
|
DisplayState *s = con->ds;
|
|
|
|
DisplayChangeListener *dcl;
|
2018-02-20 19:04:31 +08:00
|
|
|
|
2021-01-26 04:00:30 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2022-03-26 00:12:16 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-01-26 04:00:30 +08:00
|
|
|
if (dcl->ops->dpy_gl_cursor_position) {
|
|
|
|
dcl->ops->dpy_gl_cursor_position(dcl, pos_x, pos_y);
|
|
|
|
}
|
2017-10-10 21:54:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void dpy_gl_release_dmabuf(QemuConsole *con,
|
|
|
|
QemuDmaBuf *dmabuf)
|
|
|
|
{
|
2021-01-26 04:00:30 +08:00
|
|
|
DisplayState *s = con->ds;
|
|
|
|
DisplayChangeListener *dcl;
|
2017-10-10 21:54:48 +08:00
|
|
|
|
2021-01-26 04:00:30 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2022-03-26 00:12:16 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2021-01-26 04:00:30 +08:00
|
|
|
if (dcl->ops->dpy_gl_release_dmabuf) {
|
|
|
|
dcl->ops->dpy_gl_release_dmabuf(dcl, dmabuf);
|
|
|
|
}
|
2017-10-10 21:54:48 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 19:56:51 +08:00
|
|
|
void dpy_gl_update(QemuConsole *con,
|
|
|
|
uint32_t x, uint32_t y, uint32_t w, uint32_t h)
|
|
|
|
{
|
2021-01-26 04:00:30 +08:00
|
|
|
DisplayState *s = con->ds;
|
|
|
|
DisplayChangeListener *dcl;
|
|
|
|
|
2014-07-11 19:56:51 +08:00
|
|
|
assert(con->gl);
|
2021-03-11 16:11:37 +08:00
|
|
|
|
|
|
|
graphic_hw_gl_block(con, true);
|
2021-01-26 04:00:30 +08:00
|
|
|
QLIST_FOREACH(dcl, &s->listeners, next) {
|
2022-03-26 00:12:16 +08:00
|
|
|
if (con != (dcl->con ? dcl->con : active_console)) {
|
|
|
|
continue;
|
|
|
|
}
|
2022-02-15 04:13:35 +08:00
|
|
|
if (dcl->ops->dpy_gl_update) {
|
|
|
|
dcl->ops->dpy_gl_update(dcl, x, y, w, h);
|
|
|
|
}
|
2021-01-26 04:00:30 +08:00
|
|
|
}
|
2021-03-11 16:11:37 +08:00
|
|
|
graphic_hw_gl_block(con, false);
|
2014-07-11 19:56:51 +08:00
|
|
|
}
|
|
|
|
|
2010-02-11 07:29:57 +08:00
|
|
|
/***********************************************************/
|
|
|
|
/* register display */
|
|
|
|
|
2013-03-08 00:08:29 +08:00
|
|
|
/* console.c internal use only */
|
|
|
|
static DisplayState *get_alloc_displaystate(void)
|
2010-02-11 07:29:57 +08:00
|
|
|
{
|
2013-03-08 00:08:29 +08:00
|
|
|
if (!display_state) {
|
|
|
|
display_state = g_new0(DisplayState, 1);
|
|
|
|
}
|
|
|
|
return display_state;
|
2010-02-11 07:29:57 +08:00
|
|
|
}
|
|
|
|
|
2013-03-08 00:08:29 +08:00
|
|
|
/*
|
|
|
|
* Called by main(), after creating QemuConsoles
|
|
|
|
* and before initializing ui (sdl/vnc/...).
|
|
|
|
*/
|
|
|
|
DisplayState *init_displaystate(void)
|
2010-02-11 07:29:57 +08:00
|
|
|
{
|
2013-06-25 16:49:31 +08:00
|
|
|
gchar *name;
|
2018-05-07 17:54:24 +08:00
|
|
|
QemuConsole *con;
|
2013-03-08 00:08:29 +08:00
|
|
|
|
2018-05-07 17:54:24 +08:00
|
|
|
QTAILQ_FOREACH(con, &consoles, next) {
|
2023-08-30 17:37:56 +08:00
|
|
|
/* Hook up into the qom tree here (not in object_new()), once
|
2013-06-25 16:49:31 +08:00
|
|
|
* all QemuConsoles are created and the order / numbering
|
|
|
|
* doesn't change any more */
|
2018-05-07 17:54:24 +08:00
|
|
|
name = g_strdup_printf("console[%d]", con->index);
|
2013-06-25 16:49:31 +08:00
|
|
|
object_property_add_child(container_get(object_get_root(), "/backend"),
|
qom: Drop parameter @errp of object_property_add() & friends
The only way object_property_add() can fail is when a property with
the same name already exists. Since our property names are all
hardcoded, failure is a programming error, and the appropriate way to
handle it is passing &error_abort.
Same for its variants, except for object_property_add_child(), which
additionally fails when the child already has a parent. Parentage is
also under program control, so this is a programming error, too.
We have a bit over 500 callers. Almost half of them pass
&error_abort, slightly fewer ignore errors, one test case handles
errors, and the remaining few callers pass them to their own callers.
The previous few commits demonstrated once again that ignoring
programming errors is a bad idea.
Of the few ones that pass on errors, several violate the Error API.
The Error ** argument must be NULL, &error_abort, &error_fatal, or a
pointer to a variable containing NULL. Passing an argument of the
latter kind twice without clearing it in between is wrong: if the
first call sets an error, it no longer points to NULL for the second
call. ich9_pm_add_properties(), sparc32_ledma_realize(),
sparc32_dma_realize(), xilinx_axidma_realize(), xilinx_enet_realize()
are wrong that way.
When the one appropriate choice of argument is &error_abort, letting
users pick the argument is a bad idea.
Drop parameter @errp and assert the preconditions instead.
There's one exception to "duplicate property name is a programming
error": the way object_property_add() implements the magic (and
undocumented) "automatic arrayification". Don't drop @errp there.
Instead, rename object_property_add() to object_property_try_add(),
and add the obvious wrapper object_property_add().
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Paolo Bonzini <pbonzini@redhat.com>
Message-Id: <20200505152926.18877-15-armbru@redhat.com>
[Two semantic rebase conflicts resolved]
2020-05-05 23:29:22 +08:00
|
|
|
name, OBJECT(con));
|
2013-06-25 16:49:31 +08:00
|
|
|
g_free(name);
|
2013-03-08 00:08:29 +08:00
|
|
|
}
|
|
|
|
|
2010-02-11 07:29:57 +08:00
|
|
|
return display_state;
|
|
|
|
}
|
|
|
|
|
2014-09-24 23:05:27 +08:00
|
|
|
void graphic_console_set_hwops(QemuConsole *con,
|
|
|
|
const GraphicHwOps *hw_ops,
|
|
|
|
void *opaque)
|
|
|
|
{
|
|
|
|
con->hw_ops = hw_ops;
|
|
|
|
con->hw = opaque;
|
|
|
|
}
|
|
|
|
|
2014-01-24 22:35:21 +08:00
|
|
|
QemuConsole *graphic_console_init(DeviceState *dev, uint32_t head,
|
2013-04-17 16:21:27 +08:00
|
|
|
const GraphicHwOps *hw_ops,
|
2013-03-05 22:24:14 +08:00
|
|
|
void *opaque)
|
2006-04-09 09:06:34 +08:00
|
|
|
{
|
2013-04-25 18:10:45 +08:00
|
|
|
static const char noinit[] =
|
|
|
|
"Guest has not initialized the display (yet).";
|
2013-03-08 00:08:29 +08:00
|
|
|
int width = 640;
|
|
|
|
int height = 480;
|
2012-09-28 19:24:17 +08:00
|
|
|
QemuConsole *s;
|
2018-03-14 01:17:29 +08:00
|
|
|
DisplaySurface *surface;
|
2009-01-17 05:13:49 +08:00
|
|
|
|
2023-08-30 17:37:57 +08:00
|
|
|
s = qemu_graphic_console_lookup_unused();
|
2018-03-14 01:17:29 +08:00
|
|
|
if (s) {
|
|
|
|
trace_console_gfx_reuse(s->index);
|
2021-02-20 20:23:03 +08:00
|
|
|
width = qemu_console_get_width(s, 0);
|
|
|
|
height = qemu_console_get_height(s, 0);
|
2018-03-14 01:17:29 +08:00
|
|
|
} else {
|
|
|
|
trace_console_gfx_new();
|
2023-08-30 17:37:56 +08:00
|
|
|
s = (QemuConsole *)object_new(TYPE_QEMU_GRAPHIC_CONSOLE);
|
2018-03-14 01:17:29 +08:00
|
|
|
}
|
2023-08-30 17:38:03 +08:00
|
|
|
QEMU_GRAPHIC_CONSOLE(s)->head = head;
|
2014-09-24 23:05:27 +08:00
|
|
|
graphic_console_set_hwops(s, hw_ops, opaque);
|
2013-04-17 16:21:27 +08:00
|
|
|
if (dev) {
|
qom: Put name parameter before value / visitor parameter
The object_property_set_FOO() setters take property name and value in
an unusual order:
void object_property_set_FOO(Object *obj, FOO_TYPE value,
const char *name, Error **errp)
Having to pass value before name feels grating. Swap them.
Same for object_property_set(), object_property_get(), and
object_property_parse().
Convert callers with this Coccinelle script:
@@
identifier fun = {
object_property_get, object_property_parse, object_property_set_str,
object_property_set_link, object_property_set_bool,
object_property_set_int, object_property_set_uint, object_property_set,
object_property_set_qobject
};
expression obj, v, name, errp;
@@
- fun(obj, v, name, errp)
+ fun(obj, name, v, errp)
Chokes on hw/arm/musicpal.c's lcd_refresh() with the unhelpful error
message "no position information". Convert that one manually.
Fails to convert hw/arm/armsse.c, because Coccinelle gets confused by
ARMSSE being used both as typedef and function-like macro there.
Convert manually.
Fails to convert hw/rx/rx-gdbsim.c, because Coccinelle gets confused
by RXCPU being used both as typedef and function-like macro there.
Convert manually. The other files using RXCPU that way don't need
conversion.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Vladimir Sementsov-Ogievskiy <vsementsov@virtuozzo.com>
Message-Id: <20200707160613.848843-27-armbru@redhat.com>
[Straightforwad conflict with commit 2336172d9b "audio: set default
value for pcspk.iobase property" resolved]
2020-07-08 00:05:54 +08:00
|
|
|
object_property_set_link(OBJECT(s), "device", OBJECT(dev),
|
2014-04-24 22:15:58 +08:00
|
|
|
&error_abort);
|
2013-04-17 16:21:27 +08:00
|
|
|
}
|
2009-01-17 03:04:14 +08:00
|
|
|
|
2021-02-25 18:13:14 +08:00
|
|
|
surface = qemu_create_placeholder_surface(width, height, noinit);
|
2018-03-14 01:17:29 +08:00
|
|
|
dpy_gfx_replace_surface(s, surface);
|
2021-03-11 15:56:58 +08:00
|
|
|
s->gl_unblock_timer = timer_new_ms(QEMU_CLOCK_REALTIME,
|
|
|
|
graphic_hw_gl_unblock_timer, s);
|
2013-03-05 22:24:14 +08:00
|
|
|
return s;
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
|
2018-03-14 01:17:29 +08:00
|
|
|
static const GraphicHwOps unused_ops = {
|
|
|
|
/* no callbacks */
|
|
|
|
};
|
|
|
|
|
|
|
|
void graphic_console_close(QemuConsole *con)
|
|
|
|
{
|
|
|
|
static const char unplugged[] =
|
|
|
|
"Guest display has been unplugged";
|
|
|
|
DisplaySurface *surface;
|
2021-02-20 20:23:03 +08:00
|
|
|
int width = qemu_console_get_width(con, 640);
|
|
|
|
int height = qemu_console_get_height(con, 480);
|
2018-03-14 01:17:29 +08:00
|
|
|
|
|
|
|
trace_console_gfx_close(con->index);
|
qom: Put name parameter before value / visitor parameter
The object_property_set_FOO() setters take property name and value in
an unusual order:
void object_property_set_FOO(Object *obj, FOO_TYPE value,
const char *name, Error **errp)
Having to pass value before name feels grating. Swap them.
Same for object_property_set(), object_property_get(), and
object_property_parse().
Convert callers with this Coccinelle script:
@@
identifier fun = {
object_property_get, object_property_parse, object_property_set_str,
object_property_set_link, object_property_set_bool,
object_property_set_int, object_property_set_uint, object_property_set,
object_property_set_qobject
};
expression obj, v, name, errp;
@@
- fun(obj, v, name, errp)
+ fun(obj, name, v, errp)
Chokes on hw/arm/musicpal.c's lcd_refresh() with the unhelpful error
message "no position information". Convert that one manually.
Fails to convert hw/arm/armsse.c, because Coccinelle gets confused by
ARMSSE being used both as typedef and function-like macro there.
Convert manually.
Fails to convert hw/rx/rx-gdbsim.c, because Coccinelle gets confused
by RXCPU being used both as typedef and function-like macro there.
Convert manually. The other files using RXCPU that way don't need
conversion.
Signed-off-by: Markus Armbruster <armbru@redhat.com>
Reviewed-by: Eric Blake <eblake@redhat.com>
Reviewed-by: Vladimir Sementsov-Ogievskiy <vsementsov@virtuozzo.com>
Message-Id: <20200707160613.848843-27-armbru@redhat.com>
[Straightforwad conflict with commit 2336172d9b "audio: set default
value for pcspk.iobase property" resolved]
2020-07-08 00:05:54 +08:00
|
|
|
object_property_set_link(OBJECT(con), "device", NULL, &error_abort);
|
2018-03-14 01:17:29 +08:00
|
|
|
graphic_console_set_hwops(con, &unused_ops, NULL);
|
|
|
|
|
|
|
|
if (con->gl) {
|
|
|
|
dpy_gl_scanout_disable(con);
|
|
|
|
}
|
2021-02-25 18:13:14 +08:00
|
|
|
surface = qemu_create_placeholder_surface(width, height, unplugged);
|
2018-03-14 01:17:29 +08:00
|
|
|
dpy_gfx_replace_surface(con, surface);
|
|
|
|
}
|
|
|
|
|
2013-03-15 22:45:54 +08:00
|
|
|
QemuConsole *qemu_console_lookup_by_index(unsigned int index)
|
|
|
|
{
|
2018-05-07 17:54:24 +08:00
|
|
|
QemuConsole *con;
|
|
|
|
|
|
|
|
QTAILQ_FOREACH(con, &consoles, next) {
|
|
|
|
if (con->index == index) {
|
|
|
|
return con;
|
|
|
|
}
|
2013-03-15 22:45:54 +08:00
|
|
|
}
|
2018-05-07 17:54:24 +08:00
|
|
|
return NULL;
|
2013-03-15 22:45:54 +08:00
|
|
|
}
|
|
|
|
|
2014-01-24 22:35:21 +08:00
|
|
|
QemuConsole *qemu_console_lookup_by_device(DeviceState *dev, uint32_t head)
|
2013-04-18 13:30:40 +08:00
|
|
|
{
|
2018-05-07 17:54:24 +08:00
|
|
|
QemuConsole *con;
|
2013-04-18 13:30:40 +08:00
|
|
|
Object *obj;
|
2014-01-24 22:35:21 +08:00
|
|
|
uint32_t h;
|
2013-04-18 13:30:40 +08:00
|
|
|
|
2018-05-07 17:54:24 +08:00
|
|
|
QTAILQ_FOREACH(con, &consoles, next) {
|
|
|
|
obj = object_property_get_link(OBJECT(con),
|
2014-04-24 22:15:58 +08:00
|
|
|
"device", &error_abort);
|
2014-01-24 22:35:21 +08:00
|
|
|
if (DEVICE(obj) != dev) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-05-07 17:54:24 +08:00
|
|
|
h = object_property_get_uint(OBJECT(con),
|
2017-06-08 00:36:32 +08:00
|
|
|
"head", &error_abort);
|
2014-01-24 22:35:21 +08:00
|
|
|
if (h != head) {
|
|
|
|
continue;
|
2013-04-18 13:30:40 +08:00
|
|
|
}
|
2018-05-07 17:54:24 +08:00
|
|
|
return con;
|
2013-04-18 13:30:40 +08:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2016-01-12 18:45:43 +08:00
|
|
|
QemuConsole *qemu_console_lookup_by_device_name(const char *device_id,
|
|
|
|
uint32_t head, Error **errp)
|
|
|
|
{
|
|
|
|
DeviceState *dev;
|
|
|
|
QemuConsole *con;
|
|
|
|
|
|
|
|
dev = qdev_find_recursive(sysbus_get_default(), device_id);
|
|
|
|
if (dev == NULL) {
|
|
|
|
error_set(errp, ERROR_CLASS_DEVICE_NOT_FOUND,
|
|
|
|
"Device '%s' not found", device_id);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
con = qemu_console_lookup_by_device(dev, head);
|
|
|
|
if (con == NULL) {
|
|
|
|
error_setg(errp, "Device %s (head %d) is not bound to a QemuConsole",
|
|
|
|
device_id, head);
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
return con;
|
|
|
|
}
|
|
|
|
|
2023-08-30 17:37:57 +08:00
|
|
|
static QemuConsole *qemu_graphic_console_lookup_unused(void)
|
2018-03-14 01:17:29 +08:00
|
|
|
{
|
2018-05-07 17:54:24 +08:00
|
|
|
QemuConsole *con;
|
2018-03-14 01:17:29 +08:00
|
|
|
Object *obj;
|
|
|
|
|
2018-05-07 17:54:24 +08:00
|
|
|
QTAILQ_FOREACH(con, &consoles, next) {
|
2023-08-30 17:37:57 +08:00
|
|
|
if (!QEMU_IS_GRAPHIC_CONSOLE(con) || con->hw_ops != &unused_ops) {
|
2018-03-14 01:17:29 +08:00
|
|
|
continue;
|
|
|
|
}
|
2018-05-07 17:54:24 +08:00
|
|
|
obj = object_property_get_link(OBJECT(con),
|
2018-03-14 01:17:29 +08:00
|
|
|
"device", &error_abort);
|
|
|
|
if (obj != NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
2018-05-07 17:54:24 +08:00
|
|
|
return con;
|
2018-03-14 01:17:29 +08:00
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2023-01-17 19:24:40 +08:00
|
|
|
QEMUCursor *qemu_console_get_cursor(QemuConsole *con)
|
|
|
|
{
|
2023-03-19 19:10:17 +08:00
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2023-08-30 17:38:03 +08:00
|
|
|
return QEMU_IS_GRAPHIC_CONSOLE(con) ? QEMU_GRAPHIC_CONSOLE(con)->cursor : NULL;
|
2023-01-17 19:24:40 +08:00
|
|
|
}
|
|
|
|
|
2013-03-14 21:27:08 +08:00
|
|
|
bool qemu_console_is_visible(QemuConsole *con)
|
2004-07-15 01:28:59 +08:00
|
|
|
{
|
2013-03-15 22:45:54 +08:00
|
|
|
return (con == active_console) || (con->dcls > 0);
|
2004-07-15 01:28:59 +08:00
|
|
|
}
|
|
|
|
|
2013-03-14 21:27:08 +08:00
|
|
|
bool qemu_console_is_graphic(QemuConsole *con)
|
2008-09-24 11:32:33 +08:00
|
|
|
{
|
2013-03-14 21:27:08 +08:00
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2023-08-30 17:37:54 +08:00
|
|
|
return con && QEMU_IS_GRAPHIC_CONSOLE(con);
|
2013-03-14 21:27:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool qemu_console_is_fixedsize(QemuConsole *con)
|
|
|
|
{
|
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2023-08-30 17:37:54 +08:00
|
|
|
return con && (QEMU_IS_GRAPHIC_CONSOLE(con) || QEMU_IS_FIXED_TEXT_CONSOLE(con));
|
2008-09-24 11:32:33 +08:00
|
|
|
}
|
|
|
|
|
2016-09-23 15:50:27 +08:00
|
|
|
bool qemu_console_is_gl_blocked(QemuConsole *con)
|
|
|
|
{
|
|
|
|
assert(con != NULL);
|
|
|
|
return con->gl_block;
|
|
|
|
}
|
|
|
|
|
2022-06-15 14:35:14 +08:00
|
|
|
bool qemu_console_is_multihead(DeviceState *dev)
|
|
|
|
{
|
|
|
|
QemuConsole *con;
|
|
|
|
Object *obj;
|
|
|
|
uint32_t f = 0xffffffff;
|
|
|
|
uint32_t h;
|
|
|
|
|
|
|
|
QTAILQ_FOREACH(con, &consoles, next) {
|
|
|
|
obj = object_property_get_link(OBJECT(con),
|
|
|
|
"device", &error_abort);
|
|
|
|
if (DEVICE(obj) != dev) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
h = object_property_get_uint(OBJECT(con),
|
|
|
|
"head", &error_abort);
|
|
|
|
if (f == 0xffffffff) {
|
|
|
|
f = h;
|
|
|
|
} else if (h != f) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-02-17 17:41:08 +08:00
|
|
|
char *qemu_console_get_label(QemuConsole *con)
|
|
|
|
{
|
2023-08-30 17:37:54 +08:00
|
|
|
if (QEMU_IS_GRAPHIC_CONSOLE(con)) {
|
2023-08-30 17:38:03 +08:00
|
|
|
QemuGraphicConsole *c = QEMU_GRAPHIC_CONSOLE(con);
|
|
|
|
if (c->device) {
|
2022-06-15 14:35:14 +08:00
|
|
|
DeviceState *dev;
|
|
|
|
bool multihead;
|
|
|
|
|
2023-08-30 17:38:03 +08:00
|
|
|
dev = DEVICE(c->device);
|
2022-06-15 14:35:14 +08:00
|
|
|
multihead = qemu_console_is_multihead(dev);
|
|
|
|
if (multihead) {
|
|
|
|
return g_strdup_printf("%s.%d", dev->id ?
|
|
|
|
dev->id :
|
2023-08-30 17:38:03 +08:00
|
|
|
object_get_typename(c->device),
|
|
|
|
c->head);
|
2022-06-15 14:35:14 +08:00
|
|
|
} else {
|
|
|
|
return g_strdup_printf("%s", dev->id ?
|
|
|
|
dev->id :
|
2023-08-30 17:38:03 +08:00
|
|
|
object_get_typename(c->device));
|
2022-06-15 14:35:14 +08:00
|
|
|
}
|
2015-02-17 17:41:08 +08:00
|
|
|
}
|
|
|
|
return g_strdup("VGA");
|
2023-08-30 17:38:02 +08:00
|
|
|
} else if (QEMU_IS_TEXT_CONSOLE(con)) {
|
|
|
|
QemuTextConsole *c = QEMU_TEXT_CONSOLE(con);
|
|
|
|
if (c->chr && c->chr->label) {
|
|
|
|
return g_strdup(c->chr->label);
|
2015-02-17 17:41:08 +08:00
|
|
|
}
|
|
|
|
}
|
2023-08-30 17:38:02 +08:00
|
|
|
|
|
|
|
return g_strdup_printf("vc%d", con->index);
|
2015-02-17 17:41:08 +08:00
|
|
|
}
|
|
|
|
|
2013-11-28 16:58:18 +08:00
|
|
|
int qemu_console_get_index(QemuConsole *con)
|
|
|
|
{
|
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
|
|
|
return con ? con->index : -1;
|
|
|
|
}
|
|
|
|
|
2014-01-24 22:35:21 +08:00
|
|
|
uint32_t qemu_console_get_head(QemuConsole *con)
|
|
|
|
{
|
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2023-08-30 17:38:03 +08:00
|
|
|
if (con == NULL) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
if (QEMU_IS_GRAPHIC_CONSOLE(con)) {
|
|
|
|
return QEMU_GRAPHIC_CONSOLE(con)->head;
|
|
|
|
}
|
|
|
|
return 0;
|
2014-01-24 22:35:21 +08:00
|
|
|
}
|
|
|
|
|
2013-11-28 16:58:18 +08:00
|
|
|
int qemu_console_get_width(QemuConsole *con, int fallback)
|
|
|
|
{
|
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2021-02-20 20:23:03 +08:00
|
|
|
if (con == NULL) {
|
|
|
|
return fallback;
|
|
|
|
}
|
|
|
|
switch (con->scanout.kind) {
|
|
|
|
case SCANOUT_DMABUF:
|
|
|
|
return con->scanout.dmabuf->width;
|
|
|
|
case SCANOUT_TEXTURE:
|
|
|
|
return con->scanout.texture.width;
|
|
|
|
case SCANOUT_SURFACE:
|
|
|
|
return surface_width(con->surface);
|
|
|
|
default:
|
|
|
|
return fallback;
|
|
|
|
}
|
2013-11-28 16:58:18 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int qemu_console_get_height(QemuConsole *con, int fallback)
|
|
|
|
{
|
|
|
|
if (con == NULL) {
|
|
|
|
con = active_console;
|
|
|
|
}
|
2021-02-20 20:23:03 +08:00
|
|
|
if (con == NULL) {
|
|
|
|
return fallback;
|
|
|
|
}
|
|
|
|
switch (con->scanout.kind) {
|
|
|
|
case SCANOUT_DMABUF:
|
|
|
|
return con->scanout.dmabuf->height;
|
|
|
|
case SCANOUT_TEXTURE:
|
|
|
|
return con->scanout.texture.height;
|
|
|
|
case SCANOUT_SURFACE:
|
|
|
|
return surface_height(con->surface);
|
|
|
|
default:
|
|
|
|
return fallback;
|
|
|
|
}
|
2013-11-28 16:58:18 +08:00
|
|
|
}
|
|
|
|
|
2021-09-17 03:22:37 +08:00
|
|
|
static void vc_chr_accept_input(Chardev *chr)
|
|
|
|
{
|
|
|
|
VCChardev *drv = VC_CHARDEV(chr);
|
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
kbd_send_chars(drv->console);
|
2021-09-17 03:22:37 +08:00
|
|
|
}
|
|
|
|
|
2017-01-06 00:30:29 +08:00
|
|
|
static void vc_chr_set_echo(Chardev *chr, bool echo)
|
2010-12-23 20:42:52 +08:00
|
|
|
{
|
2016-12-07 23:39:10 +08:00
|
|
|
VCChardev *drv = VC_CHARDEV(chr);
|
2010-12-23 20:42:52 +08:00
|
|
|
|
2023-08-30 17:38:02 +08:00
|
|
|
drv->console->echo = echo;
|
2010-12-23 20:42:52 +08:00
|
|
|
}
|
|
|
|
|
2012-07-11 04:00:55 +08:00
|
|
|
static void text_console_update_cursor(void *opaque)
|
|
|
|
{
|
2014-05-22 17:27:13 +08:00
|
|
|
QemuConsole *s;
|
2018-05-07 17:54:24 +08:00
|
|
|
int count = 0;
|
2014-05-22 17:27:13 +08:00
|
|
|
|
|
|
|
cursor_visible_phase = !cursor_visible_phase;
|
2012-07-11 04:00:55 +08:00
|
|
|
|
2018-05-07 17:54:24 +08:00
|
|
|
QTAILQ_FOREACH(s, &consoles, next) {
|
2014-05-22 17:27:13 +08:00
|
|
|
if (qemu_console_is_graphic(s) ||
|
|
|
|
!qemu_console_is_visible(s)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
count++;
|
|
|
|
graphic_hw_invalidate(s);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (count) {
|
2023-08-30 17:38:13 +08:00
|
|
|
timer_mod(cursor_timer,
|
|
|
|
qemu_clock_get_ms(QEMU_CLOCK_REALTIME) + CONSOLE_CURSOR_PERIOD / 2);
|
2014-05-22 17:27:13 +08:00
|
|
|
}
|
2012-07-11 04:00:55 +08:00
|
|
|
}
|
|
|
|
|
2016-12-07 23:39:10 +08:00
|
|
|
static void vc_chr_open(Chardev *chr,
|
|
|
|
ChardevBackend *backend,
|
|
|
|
bool *be_opened,
|
|
|
|
Error **errp)
|
2009-01-17 04:23:27 +08:00
|
|
|
{
|
2016-12-07 20:13:50 +08:00
|
|
|
ChardevVC *vc = backend->u.vc.data;
|
2016-12-07 23:39:10 +08:00
|
|
|
VCChardev *drv = VC_CHARDEV(chr);
|
2023-08-30 17:38:02 +08:00
|
|
|
QemuTextConsole *s;
|
2013-02-25 22:52:32 +08:00
|
|
|
unsigned width = 0;
|
|
|
|
unsigned height = 0;
|
2009-01-17 04:23:27 +08:00
|
|
|
|
2013-02-25 22:52:32 +08:00
|
|
|
if (vc->has_width) {
|
|
|
|
width = vc->width;
|
|
|
|
} else if (vc->has_cols) {
|
|
|
|
width = vc->cols * FONT_WIDTH;
|
|
|
|
}
|
2010-12-23 20:42:51 +08:00
|
|
|
|
2013-02-25 22:52:32 +08:00
|
|
|
if (vc->has_height) {
|
|
|
|
height = vc->height;
|
|
|
|
} else if (vc->has_rows) {
|
|
|
|
height = vc->rows * FONT_HEIGHT;
|
|
|
|
}
|
2010-12-23 20:42:51 +08:00
|
|
|
|
2013-03-07 23:04:52 +08:00
|
|
|
trace_console_txt_new(width, height);
|
2010-12-23 20:42:51 +08:00
|
|
|
if (width == 0 || height == 0) {
|
2023-08-30 17:38:02 +08:00
|
|
|
s = QEMU_TEXT_CONSOLE(object_new(TYPE_QEMU_TEXT_CONSOLE));
|
2023-08-30 17:38:07 +08:00
|
|
|
width = qemu_console_get_width(NULL, 80 * FONT_WIDTH);
|
|
|
|
height = qemu_console_get_height(NULL, 24 * FONT_HEIGHT);
|
2010-12-23 20:42:51 +08:00
|
|
|
} else {
|
2023-08-30 17:38:02 +08:00
|
|
|
s = QEMU_TEXT_CONSOLE(object_new(TYPE_QEMU_FIXED_TEXT_CONSOLE));
|
2010-12-23 20:42:51 +08:00
|
|
|
}
|
|
|
|
|
2023-08-30 17:38:08 +08:00
|
|
|
dpy_gfx_replace_surface(QEMU_CONSOLE(s), qemu_create_displaysurface(width, height));
|
|
|
|
|
2010-12-23 20:42:51 +08:00
|
|
|
s->chr = chr;
|
2016-10-22 04:44:44 +08:00
|
|
|
drv->console = s;
|
2013-03-08 00:08:29 +08:00
|
|
|
|
2023-08-30 17:38:04 +08:00
|
|
|
/* set current text attributes to default */
|
|
|
|
drv->t_attrib = TEXT_ATTRIBUTES_DEFAULT;
|
|
|
|
text_console_resize(s);
|
|
|
|
|
|
|
|
if (chr->label) {
|
|
|
|
char *msg;
|
|
|
|
|
|
|
|
drv->t_attrib.bgcol = QEMU_COLOR_BLUE;
|
|
|
|
msg = g_strdup_printf("%s console\r\n", chr->label);
|
|
|
|
qemu_chr_write(chr, (uint8_t *)msg, strlen(msg), true);
|
|
|
|
g_free(msg);
|
|
|
|
drv->t_attrib = TEXT_ATTRIBUTES_DEFAULT;
|
|
|
|
}
|
|
|
|
|
|
|
|
*be_opened = true;
|
2013-02-20 21:43:19 +08:00
|
|
|
}
|
|
|
|
|
2013-03-05 22:24:14 +08:00
|
|
|
void qemu_console_resize(QemuConsole *s, int width, int height)
|
2008-07-02 00:24:38 +08:00
|
|
|
{
|
2022-07-25 19:58:15 +08:00
|
|
|
DisplaySurface *surface = qemu_console_surface(s);
|
2013-03-12 21:39:22 +08:00
|
|
|
|
2023-08-30 17:37:54 +08:00
|
|
|
assert(QEMU_IS_GRAPHIC_CONSOLE(s));
|
2016-08-26 17:47:11 +08:00
|
|
|
|
2022-07-25 19:58:15 +08:00
|
|
|
if ((s->scanout.kind != SCANOUT_SURFACE ||
|
|
|
|
(surface && surface->flags & QEMU_ALLOCATED_FLAG)) &&
|
|
|
|
qemu_console_get_width(s, -1) == width &&
|
2022-02-15 04:13:37 +08:00
|
|
|
qemu_console_get_height(s, -1) == height) {
|
2016-08-26 17:47:11 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-03-12 21:39:22 +08:00
|
|
|
surface = qemu_create_displaysurface(width, height);
|
|
|
|
dpy_gfx_replace_surface(s, surface);
|
2008-07-02 00:24:38 +08:00
|
|
|
}
|
2008-09-24 10:21:24 +08:00
|
|
|
|
2013-03-05 22:24:14 +08:00
|
|
|
DisplaySurface *qemu_console_surface(QemuConsole *console)
|
|
|
|
{
|
2021-02-20 20:23:03 +08:00
|
|
|
switch (console->scanout.kind) {
|
|
|
|
case SCANOUT_SURFACE:
|
|
|
|
return console->surface;
|
|
|
|
default:
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-03-05 22:24:14 +08:00
|
|
|
}
|
|
|
|
|
2009-01-24 03:56:19 +08:00
|
|
|
PixelFormat qemu_default_pixelformat(int bpp)
|
|
|
|
{
|
2014-06-18 17:07:50 +08:00
|
|
|
pixman_format_code_t fmt = qemu_default_pixman_format(bpp, true);
|
|
|
|
PixelFormat pf = qemu_pixelformat_from_pixman(fmt);
|
2009-01-16 06:14:11 +08:00
|
|
|
return pf;
|
|
|
|
}
|
2013-03-06 01:51:32 +08:00
|
|
|
|
2018-03-01 18:05:35 +08:00
|
|
|
static QemuDisplay *dpys[DISPLAY_TYPE__MAX];
|
|
|
|
|
|
|
|
void qemu_display_register(QemuDisplay *ui)
|
|
|
|
{
|
|
|
|
assert(ui->type < DISPLAY_TYPE__MAX);
|
|
|
|
dpys[ui->type] = ui;
|
|
|
|
}
|
|
|
|
|
2018-03-01 18:05:40 +08:00
|
|
|
bool qemu_display_find_default(DisplayOptions *opts)
|
|
|
|
{
|
|
|
|
static DisplayType prio[] = {
|
ui: Make the DisplayType enum entries conditional
Libvirt's "domcapabilities" command has a way to state whether certain
graphic frontends are available in QEMU or not. Originally, libvirt
looked at the "--help" output of the QEMU binary to determine whether
SDL was available or not (by looking for the "-sdl" parameter in the
help text), but since libvirt stopped doing this analysis of the help
text, the detection of SDL is currently broken, see:
https://bugzilla.redhat.com/show_bug.cgi?id=1790902
QEMU should provide a way via the QMP interface instead. A simple way,
without introducing additional commands, is to make the DisplayType
enum entries conditional, so that the enum only contains the entries if
the corresponding CONFIG_xxx switches have been set. This of course
only gives an indication which possibilities have been enabled during
compile-time of QEMU (and does not take into account whether modules
are later available or not for example - for this we'd need a separate
command), but anyway, this should already be good enough for the above
bug ticket, and it's a good idea anyway to make the QMP interface
conditional here, so let's simply do it.
Signed-off-by: Thomas Huth <thuth@redhat.com>
Message-Id: <20210615090439.70926-1-thuth@redhat.com>
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
2021-06-15 17:04:39 +08:00
|
|
|
#if defined(CONFIG_GTK)
|
2018-03-01 18:05:40 +08:00
|
|
|
DISPLAY_TYPE_GTK,
|
ui: Make the DisplayType enum entries conditional
Libvirt's "domcapabilities" command has a way to state whether certain
graphic frontends are available in QEMU or not. Originally, libvirt
looked at the "--help" output of the QEMU binary to determine whether
SDL was available or not (by looking for the "-sdl" parameter in the
help text), but since libvirt stopped doing this analysis of the help
text, the detection of SDL is currently broken, see:
https://bugzilla.redhat.com/show_bug.cgi?id=1790902
QEMU should provide a way via the QMP interface instead. A simple way,
without introducing additional commands, is to make the DisplayType
enum entries conditional, so that the enum only contains the entries if
the corresponding CONFIG_xxx switches have been set. This of course
only gives an indication which possibilities have been enabled during
compile-time of QEMU (and does not take into account whether modules
are later available or not for example - for this we'd need a separate
command), but anyway, this should already be good enough for the above
bug ticket, and it's a good idea anyway to make the QMP interface
conditional here, so let's simply do it.
Signed-off-by: Thomas Huth <thuth@redhat.com>
Message-Id: <20210615090439.70926-1-thuth@redhat.com>
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
2021-06-15 17:04:39 +08:00
|
|
|
#endif
|
|
|
|
#if defined(CONFIG_SDL)
|
2018-03-01 18:05:40 +08:00
|
|
|
DISPLAY_TYPE_SDL,
|
ui: Make the DisplayType enum entries conditional
Libvirt's "domcapabilities" command has a way to state whether certain
graphic frontends are available in QEMU or not. Originally, libvirt
looked at the "--help" output of the QEMU binary to determine whether
SDL was available or not (by looking for the "-sdl" parameter in the
help text), but since libvirt stopped doing this analysis of the help
text, the detection of SDL is currently broken, see:
https://bugzilla.redhat.com/show_bug.cgi?id=1790902
QEMU should provide a way via the QMP interface instead. A simple way,
without introducing additional commands, is to make the DisplayType
enum entries conditional, so that the enum only contains the entries if
the corresponding CONFIG_xxx switches have been set. This of course
only gives an indication which possibilities have been enabled during
compile-time of QEMU (and does not take into account whether modules
are later available or not for example - for this we'd need a separate
command), but anyway, this should already be good enough for the above
bug ticket, and it's a good idea anyway to make the QMP interface
conditional here, so let's simply do it.
Signed-off-by: Thomas Huth <thuth@redhat.com>
Message-Id: <20210615090439.70926-1-thuth@redhat.com>
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
2021-06-15 17:04:39 +08:00
|
|
|
#endif
|
|
|
|
#if defined(CONFIG_COCOA)
|
2018-03-01 18:05:40 +08:00
|
|
|
DISPLAY_TYPE_COCOA
|
ui: Make the DisplayType enum entries conditional
Libvirt's "domcapabilities" command has a way to state whether certain
graphic frontends are available in QEMU or not. Originally, libvirt
looked at the "--help" output of the QEMU binary to determine whether
SDL was available or not (by looking for the "-sdl" parameter in the
help text), but since libvirt stopped doing this analysis of the help
text, the detection of SDL is currently broken, see:
https://bugzilla.redhat.com/show_bug.cgi?id=1790902
QEMU should provide a way via the QMP interface instead. A simple way,
without introducing additional commands, is to make the DisplayType
enum entries conditional, so that the enum only contains the entries if
the corresponding CONFIG_xxx switches have been set. This of course
only gives an indication which possibilities have been enabled during
compile-time of QEMU (and does not take into account whether modules
are later available or not for example - for this we'd need a separate
command), but anyway, this should already be good enough for the above
bug ticket, and it's a good idea anyway to make the QMP interface
conditional here, so let's simply do it.
Signed-off-by: Thomas Huth <thuth@redhat.com>
Message-Id: <20210615090439.70926-1-thuth@redhat.com>
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
2021-06-15 17:04:39 +08:00
|
|
|
#endif
|
2018-03-01 18:05:40 +08:00
|
|
|
};
|
|
|
|
int i;
|
|
|
|
|
ui: Make the DisplayType enum entries conditional
Libvirt's "domcapabilities" command has a way to state whether certain
graphic frontends are available in QEMU or not. Originally, libvirt
looked at the "--help" output of the QEMU binary to determine whether
SDL was available or not (by looking for the "-sdl" parameter in the
help text), but since libvirt stopped doing this analysis of the help
text, the detection of SDL is currently broken, see:
https://bugzilla.redhat.com/show_bug.cgi?id=1790902
QEMU should provide a way via the QMP interface instead. A simple way,
without introducing additional commands, is to make the DisplayType
enum entries conditional, so that the enum only contains the entries if
the corresponding CONFIG_xxx switches have been set. This of course
only gives an indication which possibilities have been enabled during
compile-time of QEMU (and does not take into account whether modules
are later available or not for example - for this we'd need a separate
command), but anyway, this should already be good enough for the above
bug ticket, and it's a good idea anyway to make the QMP interface
conditional here, so let's simply do it.
Signed-off-by: Thomas Huth <thuth@redhat.com>
Message-Id: <20210615090439.70926-1-thuth@redhat.com>
Signed-off-by: Gerd Hoffmann <kraxel@redhat.com>
2021-06-15 17:04:39 +08:00
|
|
|
for (i = 0; i < (int)ARRAY_SIZE(prio); i++) {
|
2018-03-01 18:05:41 +08:00
|
|
|
if (dpys[prio[i]] == NULL) {
|
module: add Error arguments to module_load and module_load_qom
improve error handling during module load, by changing:
bool module_load(const char *prefix, const char *lib_name);
void module_load_qom(const char *type);
to:
int module_load(const char *prefix, const char *name, Error **errp);
int module_load_qom(const char *type, Error **errp);
where the return value is:
-1 on module load error, and errp is set with the error
0 on module or one of its dependencies are not installed
1 on module load success
2 on module load success (module already loaded or built-in)
module_load_qom_one has been introduced in:
commit 28457744c345 ("module: qom module support"), which built on top of
module_load_one, but discarded the bool return value. Restore it.
Adapt all callers to emit errors, or ignore them, or fail hard,
as appropriate in each context.
Replace the previous emission of errors via fprintf in _some_ error
conditions with Error and error_report, so as to emit to the appropriate
target.
A memory leak is also fixed as part of the module_load changes.
audio: when attempting to load an audio module, report module load errors.
Note that still for some callers, a single issue may generate multiple
error reports, and this could be improved further.
Regarding the audio code itself, audio_add() seems to ignore errors,
and this should probably be improved.
block: when attempting to load a block module, report module load errors.
For the code paths that already use the Error API, take advantage of those
to report module load errors into the Error parameter.
For the other code paths, we currently emit the error, but this could be
improved further by adding Error parameters to all possible code paths.
console: when attempting to load a display module, report module load errors.
qdev: when creating a new qdev Device object (DeviceState), report load errors.
If a module cannot be loaded to create that device, now abort execution
(if no CONFIG_MODULE) or exit (if CONFIG_MODULE).
qom/object.c: when initializing a QOM object, or looking up class_by_name,
report module load errors.
qtest: when processing the "module_load" qtest command, report errors
in the load of the module.
Signed-off-by: Claudio Fontana <cfontana@suse.de>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-Id: <20220929093035.4231-4-cfontana@suse.de>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2022-09-29 17:30:33 +08:00
|
|
|
Error *local_err = NULL;
|
|
|
|
int rv = ui_module_load(DisplayType_str(prio[i]), &local_err);
|
|
|
|
if (rv < 0) {
|
|
|
|
error_report_err(local_err);
|
|
|
|
}
|
2018-03-01 18:05:41 +08:00
|
|
|
}
|
2018-03-01 18:05:40 +08:00
|
|
|
if (dpys[prio[i]] == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
opts->type = prio[i];
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-01 18:05:35 +08:00
|
|
|
void qemu_display_early_init(DisplayOptions *opts)
|
|
|
|
{
|
|
|
|
assert(opts->type < DISPLAY_TYPE__MAX);
|
|
|
|
if (opts->type == DISPLAY_TYPE_NONE) {
|
|
|
|
return;
|
|
|
|
}
|
2018-03-01 18:05:41 +08:00
|
|
|
if (dpys[opts->type] == NULL) {
|
module: add Error arguments to module_load and module_load_qom
improve error handling during module load, by changing:
bool module_load(const char *prefix, const char *lib_name);
void module_load_qom(const char *type);
to:
int module_load(const char *prefix, const char *name, Error **errp);
int module_load_qom(const char *type, Error **errp);
where the return value is:
-1 on module load error, and errp is set with the error
0 on module or one of its dependencies are not installed
1 on module load success
2 on module load success (module already loaded or built-in)
module_load_qom_one has been introduced in:
commit 28457744c345 ("module: qom module support"), which built on top of
module_load_one, but discarded the bool return value. Restore it.
Adapt all callers to emit errors, or ignore them, or fail hard,
as appropriate in each context.
Replace the previous emission of errors via fprintf in _some_ error
conditions with Error and error_report, so as to emit to the appropriate
target.
A memory leak is also fixed as part of the module_load changes.
audio: when attempting to load an audio module, report module load errors.
Note that still for some callers, a single issue may generate multiple
error reports, and this could be improved further.
Regarding the audio code itself, audio_add() seems to ignore errors,
and this should probably be improved.
block: when attempting to load a block module, report module load errors.
For the code paths that already use the Error API, take advantage of those
to report module load errors into the Error parameter.
For the other code paths, we currently emit the error, but this could be
improved further by adding Error parameters to all possible code paths.
console: when attempting to load a display module, report module load errors.
qdev: when creating a new qdev Device object (DeviceState), report load errors.
If a module cannot be loaded to create that device, now abort execution
(if no CONFIG_MODULE) or exit (if CONFIG_MODULE).
qom/object.c: when initializing a QOM object, or looking up class_by_name,
report module load errors.
qtest: when processing the "module_load" qtest command, report errors
in the load of the module.
Signed-off-by: Claudio Fontana <cfontana@suse.de>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-Id: <20220929093035.4231-4-cfontana@suse.de>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2022-09-29 17:30:33 +08:00
|
|
|
Error *local_err = NULL;
|
|
|
|
int rv = ui_module_load(DisplayType_str(opts->type), &local_err);
|
|
|
|
if (rv < 0) {
|
|
|
|
error_report_err(local_err);
|
|
|
|
}
|
2018-03-01 18:05:41 +08:00
|
|
|
}
|
2018-03-01 18:05:35 +08:00
|
|
|
if (dpys[opts->type] == NULL) {
|
|
|
|
error_report("Display '%s' is not available.",
|
2018-08-01 17:25:08 +08:00
|
|
|
DisplayType_str(opts->type));
|
2018-03-01 18:05:35 +08:00
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
if (dpys[opts->type]->early_init) {
|
|
|
|
dpys[opts->type]->early_init(opts);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void qemu_display_init(DisplayState *ds, DisplayOptions *opts)
|
|
|
|
{
|
|
|
|
assert(opts->type < DISPLAY_TYPE__MAX);
|
|
|
|
if (opts->type == DISPLAY_TYPE_NONE) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
assert(dpys[opts->type] != NULL);
|
|
|
|
dpys[opts->type]->init(ds, opts);
|
|
|
|
}
|
|
|
|
|
2020-01-08 22:47:02 +08:00
|
|
|
void qemu_display_help(void)
|
|
|
|
{
|
|
|
|
int idx;
|
|
|
|
|
|
|
|
printf("Available display backend types:\n");
|
2020-01-21 03:29:47 +08:00
|
|
|
printf("none\n");
|
2020-01-08 22:47:02 +08:00
|
|
|
for (idx = DISPLAY_TYPE_NONE; idx < DISPLAY_TYPE__MAX; idx++) {
|
|
|
|
if (!dpys[idx]) {
|
module: add Error arguments to module_load and module_load_qom
improve error handling during module load, by changing:
bool module_load(const char *prefix, const char *lib_name);
void module_load_qom(const char *type);
to:
int module_load(const char *prefix, const char *name, Error **errp);
int module_load_qom(const char *type, Error **errp);
where the return value is:
-1 on module load error, and errp is set with the error
0 on module or one of its dependencies are not installed
1 on module load success
2 on module load success (module already loaded or built-in)
module_load_qom_one has been introduced in:
commit 28457744c345 ("module: qom module support"), which built on top of
module_load_one, but discarded the bool return value. Restore it.
Adapt all callers to emit errors, or ignore them, or fail hard,
as appropriate in each context.
Replace the previous emission of errors via fprintf in _some_ error
conditions with Error and error_report, so as to emit to the appropriate
target.
A memory leak is also fixed as part of the module_load changes.
audio: when attempting to load an audio module, report module load errors.
Note that still for some callers, a single issue may generate multiple
error reports, and this could be improved further.
Regarding the audio code itself, audio_add() seems to ignore errors,
and this should probably be improved.
block: when attempting to load a block module, report module load errors.
For the code paths that already use the Error API, take advantage of those
to report module load errors into the Error parameter.
For the other code paths, we currently emit the error, but this could be
improved further by adding Error parameters to all possible code paths.
console: when attempting to load a display module, report module load errors.
qdev: when creating a new qdev Device object (DeviceState), report load errors.
If a module cannot be loaded to create that device, now abort execution
(if no CONFIG_MODULE) or exit (if CONFIG_MODULE).
qom/object.c: when initializing a QOM object, or looking up class_by_name,
report module load errors.
qtest: when processing the "module_load" qtest command, report errors
in the load of the module.
Signed-off-by: Claudio Fontana <cfontana@suse.de>
Reviewed-by: Richard Henderson <richard.henderson@linaro.org>
Message-Id: <20220929093035.4231-4-cfontana@suse.de>
Signed-off-by: Paolo Bonzini <pbonzini@redhat.com>
2022-09-29 17:30:33 +08:00
|
|
|
Error *local_err = NULL;
|
|
|
|
int rv = ui_module_load(DisplayType_str(idx), &local_err);
|
|
|
|
if (rv < 0) {
|
|
|
|
error_report_err(local_err);
|
|
|
|
}
|
2020-01-08 22:47:02 +08:00
|
|
|
}
|
|
|
|
if (dpys[idx]) {
|
|
|
|
printf("%s\n", DisplayType_str(dpys[idx]->type));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-07 20:13:50 +08:00
|
|
|
void qemu_chr_parse_vc(QemuOpts *opts, ChardevBackend *backend, Error **errp)
|
2013-02-25 22:52:32 +08:00
|
|
|
{
|
|
|
|
int val;
|
2016-02-20 08:19:31 +08:00
|
|
|
ChardevVC *vc;
|
2013-02-25 22:52:32 +08:00
|
|
|
|
2016-12-09 16:04:51 +08:00
|
|
|
backend->type = CHARDEV_BACKEND_KIND_VC;
|
qapi: Don't special-case simple union wrappers
Simple unions were carrying a special case that hid their 'data'
QMP member from the resulting C struct, via the hack method
QAPISchemaObjectTypeVariant.simple_union_type(). But by using
the work we started by unboxing flat union and alternate
branches, coupled with the ability to visit the members of an
implicit type, we can now expose the simple union's implicit
type in qapi-types.h:
| struct q_obj_ImageInfoSpecificQCow2_wrapper {
| ImageInfoSpecificQCow2 *data;
| };
|
| struct q_obj_ImageInfoSpecificVmdk_wrapper {
| ImageInfoSpecificVmdk *data;
| };
...
| struct ImageInfoSpecific {
| ImageInfoSpecificKind type;
| union { /* union tag is @type */
| void *data;
|- ImageInfoSpecificQCow2 *qcow2;
|- ImageInfoSpecificVmdk *vmdk;
|+ q_obj_ImageInfoSpecificQCow2_wrapper qcow2;
|+ q_obj_ImageInfoSpecificVmdk_wrapper vmdk;
| } u;
| };
Doing this removes asymmetry between QAPI's QMP side and its
C side (both sides now expose 'data'), and means that the
treatment of a simple union as sugar for a flat union is now
equivalent in both languages (previously the two approaches used
a different layer of dereferencing, where the simple union could
be converted to a flat union with equivalent C layout but
different {} on the wire, or to an equivalent QMP wire form
but with different C representation). Using the implicit type
also lets us get rid of the simple_union_type() hack.
Of course, now all clients of simple unions have to adjust from
using su->u.member to using su->u.member.data; while this touches
a number of files in the tree, some earlier cleanup patches
helped minimize the change to the initialization of a temporary
variable rather than every single member access. The generated
qapi-visit.c code is also affected by the layout change:
|@@ -7393,10 +7393,10 @@ void visit_type_ImageInfoSpecific_member
| }
| switch (obj->type) {
| case IMAGE_INFO_SPECIFIC_KIND_QCOW2:
|- visit_type_ImageInfoSpecificQCow2(v, "data", &obj->u.qcow2, &err);
|+ visit_type_q_obj_ImageInfoSpecificQCow2_wrapper_members(v, &obj->u.qcow2, &err);
| break;
| case IMAGE_INFO_SPECIFIC_KIND_VMDK:
|- visit_type_ImageInfoSpecificVmdk(v, "data", &obj->u.vmdk, &err);
|+ visit_type_q_obj_ImageInfoSpecificVmdk_wrapper_members(v, &obj->u.vmdk, &err);
| break;
| default:
| abort();
Signed-off-by: Eric Blake <eblake@redhat.com>
Message-Id: <1458254921-17042-13-git-send-email-eblake@redhat.com>
Signed-off-by: Markus Armbruster <armbru@redhat.com>
2016-03-18 06:48:37 +08:00
|
|
|
vc = backend->u.vc.data = g_new0(ChardevVC, 1);
|
2016-02-20 08:19:31 +08:00
|
|
|
qemu_chr_parse_common(opts, qapi_ChardevVC_base(vc));
|
2013-02-25 22:52:32 +08:00
|
|
|
|
|
|
|
val = qemu_opt_get_number(opts, "width", 0);
|
|
|
|
if (val != 0) {
|
2016-02-20 08:19:31 +08:00
|
|
|
vc->has_width = true;
|
|
|
|
vc->width = val;
|
2013-02-25 22:52:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
val = qemu_opt_get_number(opts, "height", 0);
|
|
|
|
if (val != 0) {
|
2016-02-20 08:19:31 +08:00
|
|
|
vc->has_height = true;
|
|
|
|
vc->height = val;
|
2013-02-25 22:52:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
val = qemu_opt_get_number(opts, "cols", 0);
|
|
|
|
if (val != 0) {
|
2016-02-20 08:19:31 +08:00
|
|
|
vc->has_cols = true;
|
|
|
|
vc->cols = val;
|
2013-02-25 22:52:32 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
val = qemu_opt_get_number(opts, "rows", 0);
|
|
|
|
if (val != 0) {
|
2016-02-20 08:19:31 +08:00
|
|
|
vc->has_rows = true;
|
|
|
|
vc->rows = val;
|
2013-02-25 22:52:32 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-07 23:39:10 +08:00
|
|
|
static void char_vc_class_init(ObjectClass *oc, void *data)
|
|
|
|
{
|
|
|
|
ChardevClass *cc = CHARDEV_CLASS(oc);
|
|
|
|
|
2016-12-09 05:50:12 +08:00
|
|
|
cc->parse = qemu_chr_parse_vc;
|
2016-12-07 23:39:10 +08:00
|
|
|
cc->open = vc_chr_open;
|
|
|
|
cc->chr_write = vc_chr_write;
|
2021-09-17 03:22:37 +08:00
|
|
|
cc->chr_accept_input = vc_chr_accept_input;
|
2016-12-07 23:39:10 +08:00
|
|
|
cc->chr_set_echo = vc_chr_set_echo;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const TypeInfo char_vc_type_info = {
|
|
|
|
.name = TYPE_CHARDEV_VC,
|
|
|
|
.parent = TYPE_CHARDEV,
|
2016-12-07 21:20:22 +08:00
|
|
|
.instance_size = sizeof(VCChardev),
|
2016-12-07 23:39:10 +08:00
|
|
|
.class_init = char_vc_class_init,
|
|
|
|
};
|
|
|
|
|
|
|
|
void qemu_console_early_init(void)
|
|
|
|
{
|
|
|
|
/* set the default vc driver */
|
|
|
|
if (!object_class_by_name(TYPE_CHARDEV_VC)) {
|
|
|
|
type_register(&char_vc_type_info);
|
|
|
|
}
|
|
|
|
}
|