virtio-gpu: add live migration support
Store some additional state for cursor and resource backing storage, so we can write out and reload things. Implement vmsave+vmload for 2d mode. Continue blocking live migration in 3d/virgl mode. Signed-off-by: Gerd Hoffmann <kraxel@redhat.com> Message-id: 1464009727-7753-1-git-send-email-kraxel@redhat.com
This commit is contained in:
parent
4e68a0ee17
commit
0c244e50ee
@ -284,7 +284,7 @@ static void virgl_resource_attach_backing(VirtIOGPU *g,
|
|||||||
VIRTIO_GPU_FILL_CMD(att_rb);
|
VIRTIO_GPU_FILL_CMD(att_rb);
|
||||||
trace_virtio_gpu_cmd_res_back_attach(att_rb.resource_id);
|
trace_virtio_gpu_cmd_res_back_attach(att_rb.resource_id);
|
||||||
|
|
||||||
ret = virtio_gpu_create_mapping_iov(&att_rb, cmd, &res_iovs);
|
ret = virtio_gpu_create_mapping_iov(&att_rb, cmd, NULL, &res_iovs);
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
cmd->error = VIRTIO_GPU_RESP_ERR_UNSPEC;
|
cmd->error = VIRTIO_GPU_RESP_ERR_UNSPEC;
|
||||||
return;
|
return;
|
||||||
|
@ -22,6 +22,8 @@
|
|||||||
#include "qemu/log.h"
|
#include "qemu/log.h"
|
||||||
#include "qapi/error.h"
|
#include "qapi/error.h"
|
||||||
|
|
||||||
|
#define VIRTIO_GPU_VM_VERSION 1
|
||||||
|
|
||||||
static struct virtio_gpu_simple_resource*
|
static struct virtio_gpu_simple_resource*
|
||||||
virtio_gpu_find_resource(VirtIOGPU *g, uint32_t resource_id);
|
virtio_gpu_find_resource(VirtIOGPU *g, uint32_t resource_id);
|
||||||
|
|
||||||
@ -94,7 +96,7 @@ static void update_cursor_data_virgl(VirtIOGPU *g,
|
|||||||
static void update_cursor(VirtIOGPU *g, struct virtio_gpu_update_cursor *cursor)
|
static void update_cursor(VirtIOGPU *g, struct virtio_gpu_update_cursor *cursor)
|
||||||
{
|
{
|
||||||
struct virtio_gpu_scanout *s;
|
struct virtio_gpu_scanout *s;
|
||||||
bool move = cursor->hdr.type != VIRTIO_GPU_CMD_MOVE_CURSOR;
|
bool move = cursor->hdr.type == VIRTIO_GPU_CMD_MOVE_CURSOR;
|
||||||
|
|
||||||
if (cursor->pos.scanout_id >= g->conf.max_outputs) {
|
if (cursor->pos.scanout_id >= g->conf.max_outputs) {
|
||||||
return;
|
return;
|
||||||
@ -107,7 +109,7 @@ static void update_cursor(VirtIOGPU *g, struct virtio_gpu_update_cursor *cursor)
|
|||||||
move ? "move" : "update",
|
move ? "move" : "update",
|
||||||
cursor->resource_id);
|
cursor->resource_id);
|
||||||
|
|
||||||
if (move) {
|
if (!move) {
|
||||||
if (!s->current_cursor) {
|
if (!s->current_cursor) {
|
||||||
s->current_cursor = cursor_alloc(64, 64);
|
s->current_cursor = cursor_alloc(64, 64);
|
||||||
}
|
}
|
||||||
@ -120,6 +122,11 @@ static void update_cursor(VirtIOGPU *g, struct virtio_gpu_update_cursor *cursor)
|
|||||||
g, s, cursor->resource_id);
|
g, s, cursor->resource_id);
|
||||||
}
|
}
|
||||||
dpy_cursor_define(s->con, s->current_cursor);
|
dpy_cursor_define(s->con, s->current_cursor);
|
||||||
|
|
||||||
|
s->cursor = *cursor;
|
||||||
|
} else {
|
||||||
|
s->cursor.pos.x = cursor->pos.x;
|
||||||
|
s->cursor.pos.y = cursor->pos.y;
|
||||||
}
|
}
|
||||||
dpy_mouse_set(s->con, cursor->pos.x, cursor->pos.y,
|
dpy_mouse_set(s->con, cursor->pos.x, cursor->pos.y,
|
||||||
cursor->resource_id ? 1 : 0);
|
cursor->resource_id ? 1 : 0);
|
||||||
@ -602,7 +609,7 @@ static void virtio_gpu_set_scanout(VirtIOGPU *g,
|
|||||||
|
|
||||||
int virtio_gpu_create_mapping_iov(struct virtio_gpu_resource_attach_backing *ab,
|
int virtio_gpu_create_mapping_iov(struct virtio_gpu_resource_attach_backing *ab,
|
||||||
struct virtio_gpu_ctrl_command *cmd,
|
struct virtio_gpu_ctrl_command *cmd,
|
||||||
struct iovec **iov)
|
uint64_t **addr, struct iovec **iov)
|
||||||
{
|
{
|
||||||
struct virtio_gpu_mem_entry *ents;
|
struct virtio_gpu_mem_entry *ents;
|
||||||
size_t esize, s;
|
size_t esize, s;
|
||||||
@ -628,10 +635,16 @@ int virtio_gpu_create_mapping_iov(struct virtio_gpu_resource_attach_backing *ab,
|
|||||||
}
|
}
|
||||||
|
|
||||||
*iov = g_malloc0(sizeof(struct iovec) * ab->nr_entries);
|
*iov = g_malloc0(sizeof(struct iovec) * ab->nr_entries);
|
||||||
|
if (addr) {
|
||||||
|
*addr = g_malloc0(sizeof(uint64_t) * ab->nr_entries);
|
||||||
|
}
|
||||||
for (i = 0; i < ab->nr_entries; i++) {
|
for (i = 0; i < ab->nr_entries; i++) {
|
||||||
hwaddr len = ents[i].length;
|
hwaddr len = ents[i].length;
|
||||||
(*iov)[i].iov_len = ents[i].length;
|
(*iov)[i].iov_len = ents[i].length;
|
||||||
(*iov)[i].iov_base = cpu_physical_memory_map(ents[i].addr, &len, 1);
|
(*iov)[i].iov_base = cpu_physical_memory_map(ents[i].addr, &len, 1);
|
||||||
|
if (addr) {
|
||||||
|
(*addr)[i] = ents[i].addr;
|
||||||
|
}
|
||||||
if (!(*iov)[i].iov_base || len != ents[i].length) {
|
if (!(*iov)[i].iov_base || len != ents[i].length) {
|
||||||
qemu_log_mask(LOG_GUEST_ERROR, "%s: failed to map MMIO memory for"
|
qemu_log_mask(LOG_GUEST_ERROR, "%s: failed to map MMIO memory for"
|
||||||
" resource %d element %d\n",
|
" resource %d element %d\n",
|
||||||
@ -639,6 +652,10 @@ int virtio_gpu_create_mapping_iov(struct virtio_gpu_resource_attach_backing *ab,
|
|||||||
virtio_gpu_cleanup_mapping_iov(*iov, i);
|
virtio_gpu_cleanup_mapping_iov(*iov, i);
|
||||||
g_free(ents);
|
g_free(ents);
|
||||||
*iov = NULL;
|
*iov = NULL;
|
||||||
|
if (addr) {
|
||||||
|
g_free(*addr);
|
||||||
|
*addr = NULL;
|
||||||
|
}
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -662,6 +679,8 @@ static void virtio_gpu_cleanup_mapping(struct virtio_gpu_simple_resource *res)
|
|||||||
virtio_gpu_cleanup_mapping_iov(res->iov, res->iov_cnt);
|
virtio_gpu_cleanup_mapping_iov(res->iov, res->iov_cnt);
|
||||||
res->iov = NULL;
|
res->iov = NULL;
|
||||||
res->iov_cnt = 0;
|
res->iov_cnt = 0;
|
||||||
|
g_free(res->addrs);
|
||||||
|
res->addrs = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
static void
|
||||||
@ -683,7 +702,7 @@ virtio_gpu_resource_attach_backing(VirtIOGPU *g,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
ret = virtio_gpu_create_mapping_iov(&ab, cmd, &res->iov);
|
ret = virtio_gpu_create_mapping_iov(&ab, cmd, &res->addrs, &res->iov);
|
||||||
if (ret != 0) {
|
if (ret != 0) {
|
||||||
cmd->error = VIRTIO_GPU_RESP_ERR_UNSPEC;
|
cmd->error = VIRTIO_GPU_RESP_ERR_UNSPEC;
|
||||||
return;
|
return;
|
||||||
@ -929,11 +948,163 @@ const GraphicHwOps virtio_gpu_ops = {
|
|||||||
.gl_block = virtio_gpu_gl_block,
|
.gl_block = virtio_gpu_gl_block,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const VMStateDescription vmstate_virtio_gpu_scanout = {
|
||||||
|
.name = "virtio-gpu-one-scanout",
|
||||||
|
.version_id = 1,
|
||||||
|
.fields = (VMStateField[]) {
|
||||||
|
VMSTATE_UINT32(resource_id, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_UINT32(width, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_UINT32(height, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_INT32(x, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_INT32(y, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_UINT32(cursor.resource_id, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_UINT32(cursor.hot_x, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_UINT32(cursor.hot_y, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_UINT32(cursor.pos.x, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_UINT32(cursor.pos.y, struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_END_OF_LIST()
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
static const VMStateDescription vmstate_virtio_gpu_scanouts = {
|
||||||
|
.name = "virtio-gpu-scanouts",
|
||||||
|
.version_id = 1,
|
||||||
|
.fields = (VMStateField[]) {
|
||||||
|
VMSTATE_INT32(enable, struct VirtIOGPU),
|
||||||
|
VMSTATE_UINT32_EQUAL(conf.max_outputs, struct VirtIOGPU),
|
||||||
|
VMSTATE_STRUCT_VARRAY_UINT32(scanout, struct VirtIOGPU,
|
||||||
|
conf.max_outputs, 1,
|
||||||
|
vmstate_virtio_gpu_scanout,
|
||||||
|
struct virtio_gpu_scanout),
|
||||||
|
VMSTATE_END_OF_LIST()
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
static const VMStateDescription vmstate_virtio_gpu_unmigratable = {
|
static const VMStateDescription vmstate_virtio_gpu_unmigratable = {
|
||||||
.name = "virtio-gpu",
|
.name = "virtio-gpu-with-virgl",
|
||||||
.unmigratable = 1,
|
.unmigratable = 1,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void virtio_gpu_save(QEMUFile *f, void *opaque)
|
||||||
|
{
|
||||||
|
VirtIOGPU *g = opaque;
|
||||||
|
VirtIODevice *vdev = VIRTIO_DEVICE(g);
|
||||||
|
struct virtio_gpu_simple_resource *res;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
virtio_save(vdev, f);
|
||||||
|
|
||||||
|
/* in 2d mode we should never find unprocessed commands here */
|
||||||
|
assert(QTAILQ_EMPTY(&g->cmdq));
|
||||||
|
|
||||||
|
QTAILQ_FOREACH(res, &g->reslist, next) {
|
||||||
|
qemu_put_be32(f, res->resource_id);
|
||||||
|
qemu_put_be32(f, res->width);
|
||||||
|
qemu_put_be32(f, res->height);
|
||||||
|
qemu_put_be32(f, res->format);
|
||||||
|
qemu_put_be32(f, res->iov_cnt);
|
||||||
|
for (i = 0; i < res->iov_cnt; i++) {
|
||||||
|
qemu_put_be64(f, res->addrs[i]);
|
||||||
|
qemu_put_be32(f, res->iov[i].iov_len);
|
||||||
|
}
|
||||||
|
qemu_put_buffer(f, (void *)pixman_image_get_data(res->image),
|
||||||
|
pixman_image_get_stride(res->image) * res->height);
|
||||||
|
}
|
||||||
|
qemu_put_be32(f, 0); /* end of list */
|
||||||
|
|
||||||
|
vmstate_save_state(f, &vmstate_virtio_gpu_scanouts, g, NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
static int virtio_gpu_load(QEMUFile *f, void *opaque, int version_id)
|
||||||
|
{
|
||||||
|
VirtIOGPU *g = opaque;
|
||||||
|
VirtIODevice *vdev = VIRTIO_DEVICE(g);
|
||||||
|
struct virtio_gpu_simple_resource *res;
|
||||||
|
struct virtio_gpu_scanout *scanout;
|
||||||
|
uint32_t resource_id, pformat;
|
||||||
|
int i, ret;
|
||||||
|
|
||||||
|
if (version_id != VIRTIO_GPU_VM_VERSION) {
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = virtio_load(vdev, f, version_id);
|
||||||
|
if (ret) {
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
resource_id = qemu_get_be32(f);
|
||||||
|
while (resource_id != 0) {
|
||||||
|
res = g_new0(struct virtio_gpu_simple_resource, 1);
|
||||||
|
res->resource_id = resource_id;
|
||||||
|
res->width = qemu_get_be32(f);
|
||||||
|
res->height = qemu_get_be32(f);
|
||||||
|
res->format = qemu_get_be32(f);
|
||||||
|
res->iov_cnt = qemu_get_be32(f);
|
||||||
|
|
||||||
|
/* allocate */
|
||||||
|
pformat = get_pixman_format(res->format);
|
||||||
|
if (!pformat) {
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
res->image = pixman_image_create_bits(pformat,
|
||||||
|
res->width, res->height,
|
||||||
|
NULL, 0);
|
||||||
|
if (!res->image) {
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
res->addrs = g_new(uint64_t, res->iov_cnt);
|
||||||
|
res->iov = g_new(struct iovec, res->iov_cnt);
|
||||||
|
|
||||||
|
/* read data */
|
||||||
|
for (i = 0; i < res->iov_cnt; i++) {
|
||||||
|
res->addrs[i] = qemu_get_be64(f);
|
||||||
|
res->iov[i].iov_len = qemu_get_be32(f);
|
||||||
|
}
|
||||||
|
qemu_get_buffer(f, (void *)pixman_image_get_data(res->image),
|
||||||
|
pixman_image_get_stride(res->image) * res->height);
|
||||||
|
|
||||||
|
/* restore mapping */
|
||||||
|
for (i = 0; i < res->iov_cnt; i++) {
|
||||||
|
hwaddr len = res->iov[i].iov_len;
|
||||||
|
res->iov[i].iov_base =
|
||||||
|
cpu_physical_memory_map(res->addrs[i], &len, 1);
|
||||||
|
if (!res->iov[i].iov_base || len != res->iov[i].iov_len) {
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QTAILQ_INSERT_HEAD(&g->reslist, res, next);
|
||||||
|
|
||||||
|
resource_id = qemu_get_be32(f);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* load & apply scanout state */
|
||||||
|
vmstate_load_state(f, &vmstate_virtio_gpu_scanouts, g, 1);
|
||||||
|
for (i = 0; i < g->conf.max_outputs; i++) {
|
||||||
|
scanout = &g->scanout[i];
|
||||||
|
if (!scanout->resource_id) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
res = virtio_gpu_find_resource(g, scanout->resource_id);
|
||||||
|
if (!res) {
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
scanout->ds = qemu_create_displaysurface_pixman(res->image);
|
||||||
|
if (!scanout->ds) {
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
dpy_gfx_replace_surface(scanout->con, scanout->ds);
|
||||||
|
dpy_gfx_update(scanout->con, 0, 0, scanout->width, scanout->height);
|
||||||
|
update_cursor(g, &scanout->cursor);
|
||||||
|
res->scanout_bitmask |= (1 << i);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static void virtio_gpu_device_realize(DeviceState *qdev, Error **errp)
|
static void virtio_gpu_device_realize(DeviceState *qdev, Error **errp)
|
||||||
{
|
{
|
||||||
VirtIODevice *vdev = VIRTIO_DEVICE(qdev);
|
VirtIODevice *vdev = VIRTIO_DEVICE(qdev);
|
||||||
@ -991,7 +1162,12 @@ static void virtio_gpu_device_realize(DeviceState *qdev, Error **errp)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
vmstate_register(qdev, -1, &vmstate_virtio_gpu_unmigratable, g);
|
if (virtio_gpu_virgl_enabled(g->conf)) {
|
||||||
|
vmstate_register(qdev, -1, &vmstate_virtio_gpu_unmigratable, g);
|
||||||
|
} else {
|
||||||
|
register_savevm(qdev, "virtio-gpu", -1, VIRTIO_GPU_VM_VERSION,
|
||||||
|
virtio_gpu_save, virtio_gpu_load, g);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void virtio_gpu_instance_init(Object *obj)
|
static void virtio_gpu_instance_init(Object *obj)
|
||||||
|
@ -84,6 +84,17 @@ static const GraphicHwOps virtio_vga_ops = {
|
|||||||
.gl_block = virtio_vga_gl_block,
|
.gl_block = virtio_vga_gl_block,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static const VMStateDescription vmstate_virtio_vga = {
|
||||||
|
.name = "virtio-vga",
|
||||||
|
.version_id = 2,
|
||||||
|
.minimum_version_id = 2,
|
||||||
|
.fields = (VMStateField[]) {
|
||||||
|
/* no pci stuff here, saving the virtio device will handle that */
|
||||||
|
VMSTATE_STRUCT(vga, VirtIOVGA, 0, vmstate_vga_common, VGACommonState),
|
||||||
|
VMSTATE_END_OF_LIST()
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
/* VGA device wrapper around PCI device around virtio GPU */
|
/* VGA device wrapper around PCI device around virtio GPU */
|
||||||
static void virtio_vga_realize(VirtIOPCIProxy *vpci_dev, Error **errp)
|
static void virtio_vga_realize(VirtIOPCIProxy *vpci_dev, Error **errp)
|
||||||
{
|
{
|
||||||
@ -168,6 +179,7 @@ static void virtio_vga_class_init(ObjectClass *klass, void *data)
|
|||||||
set_bit(DEVICE_CATEGORY_DISPLAY, dc->categories);
|
set_bit(DEVICE_CATEGORY_DISPLAY, dc->categories);
|
||||||
dc->props = virtio_vga_properties;
|
dc->props = virtio_vga_properties;
|
||||||
dc->reset = virtio_vga_reset;
|
dc->reset = virtio_vga_reset;
|
||||||
|
dc->vmsd = &vmstate_virtio_vga;
|
||||||
dc->hotpluggable = false;
|
dc->hotpluggable = false;
|
||||||
|
|
||||||
k->realize = virtio_vga_realize;
|
k->realize = virtio_vga_realize;
|
||||||
|
@ -32,6 +32,7 @@ struct virtio_gpu_simple_resource {
|
|||||||
uint32_t width;
|
uint32_t width;
|
||||||
uint32_t height;
|
uint32_t height;
|
||||||
uint32_t format;
|
uint32_t format;
|
||||||
|
uint64_t *addrs;
|
||||||
struct iovec *iov;
|
struct iovec *iov;
|
||||||
unsigned int iov_cnt;
|
unsigned int iov_cnt;
|
||||||
uint32_t scanout_bitmask;
|
uint32_t scanout_bitmask;
|
||||||
@ -46,6 +47,7 @@ struct virtio_gpu_scanout {
|
|||||||
int x, y;
|
int x, y;
|
||||||
int invalidate;
|
int invalidate;
|
||||||
uint32_t resource_id;
|
uint32_t resource_id;
|
||||||
|
struct virtio_gpu_update_cursor cursor;
|
||||||
QEMUCursor *current_cursor;
|
QEMUCursor *current_cursor;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -150,7 +152,7 @@ void virtio_gpu_get_display_info(VirtIOGPU *g,
|
|||||||
struct virtio_gpu_ctrl_command *cmd);
|
struct virtio_gpu_ctrl_command *cmd);
|
||||||
int virtio_gpu_create_mapping_iov(struct virtio_gpu_resource_attach_backing *ab,
|
int virtio_gpu_create_mapping_iov(struct virtio_gpu_resource_attach_backing *ab,
|
||||||
struct virtio_gpu_ctrl_command *cmd,
|
struct virtio_gpu_ctrl_command *cmd,
|
||||||
struct iovec **iov);
|
uint64_t **addr, struct iovec **iov);
|
||||||
void virtio_gpu_cleanup_mapping_iov(struct iovec *iov, uint32_t count);
|
void virtio_gpu_cleanup_mapping_iov(struct iovec *iov, uint32_t count);
|
||||||
void virtio_gpu_process_cmdq(VirtIOGPU *g);
|
void virtio_gpu_process_cmdq(VirtIOGPU *g);
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user