1 #include "qemu/osdep.h" 2 #include "qapi/error.h" 3 #include "include/qemu-common.h" 4 #include "chardev/char.h" 5 #include "qemu/buffer.h" 6 #include "qemu/option.h" 7 #include "qemu/units.h" 8 #include "hw/qdev-core.h" 9 #include "migration/blocker.h" 10 #include "ui/clipboard.h" 11 #include "ui/console.h" 12 #include "ui/input.h" 13 #include "trace.h" 14 15 #include "qapi/qapi-types-char.h" 16 #include "qapi/qapi-types-ui.h" 17 18 #include "spice/vd_agent.h" 19 20 #define CHECK_SPICE_PROTOCOL_VERSION(major, minor, micro) \ 21 (CONFIG_SPICE_PROTOCOL_MAJOR > (major) || \ 22 (CONFIG_SPICE_PROTOCOL_MAJOR == (major) && \ 23 CONFIG_SPICE_PROTOCOL_MINOR > (minor)) || \ 24 (CONFIG_SPICE_PROTOCOL_MAJOR == (major) && \ 25 CONFIG_SPICE_PROTOCOL_MINOR == (minor) && \ 26 CONFIG_SPICE_PROTOCOL_MICRO >= (micro))) 27 28 #define VDAGENT_BUFFER_LIMIT (1 * MiB) 29 #define VDAGENT_MOUSE_DEFAULT true 30 #define VDAGENT_CLIPBOARD_DEFAULT false 31 32 struct VDAgentChardev { 33 Chardev parent; 34 35 /* TODO: migration isn't yet supported */ 36 Error *migration_blocker; 37 38 /* config */ 39 bool mouse; 40 bool clipboard; 41 42 /* guest vdagent */ 43 uint32_t caps; 44 VDIChunkHeader chunk; 45 uint32_t chunksize; 46 uint8_t *msgbuf; 47 uint32_t msgsize; 48 uint8_t *xbuf; 49 uint32_t xoff, xsize; 50 Buffer outbuf; 51 52 /* mouse */ 53 DeviceState mouse_dev; 54 uint32_t mouse_x; 55 uint32_t mouse_y; 56 uint32_t mouse_btn; 57 uint32_t mouse_display; 58 QemuInputHandlerState *mouse_hs; 59 60 /* clipboard */ 61 QemuClipboardPeer cbpeer; 62 uint32_t last_serial[QEMU_CLIPBOARD_SELECTION__COUNT]; 63 uint32_t cbpending[QEMU_CLIPBOARD_SELECTION__COUNT]; 64 }; 65 typedef struct VDAgentChardev VDAgentChardev; 66 67 #define TYPE_CHARDEV_QEMU_VDAGENT "chardev-qemu-vdagent" 68 69 DECLARE_INSTANCE_CHECKER(VDAgentChardev, QEMU_VDAGENT_CHARDEV, 70 TYPE_CHARDEV_QEMU_VDAGENT); 71 72 /* ------------------------------------------------------------------ */ 73 /* names, for debug logging */ 74 75 static const char *cap_name[] = { 76 [VD_AGENT_CAP_MOUSE_STATE] = "mouse-state", 77 [VD_AGENT_CAP_MONITORS_CONFIG] = "monitors-config", 78 [VD_AGENT_CAP_REPLY] = "reply", 79 [VD_AGENT_CAP_CLIPBOARD] = "clipboard", 80 [VD_AGENT_CAP_DISPLAY_CONFIG] = "display-config", 81 [VD_AGENT_CAP_CLIPBOARD_BY_DEMAND] = "clipboard-by-demand", 82 [VD_AGENT_CAP_CLIPBOARD_SELECTION] = "clipboard-selection", 83 [VD_AGENT_CAP_SPARSE_MONITORS_CONFIG] = "sparse-monitors-config", 84 [VD_AGENT_CAP_GUEST_LINEEND_LF] = "guest-lineend-lf", 85 [VD_AGENT_CAP_GUEST_LINEEND_CRLF] = "guest-lineend-crlf", 86 [VD_AGENT_CAP_MAX_CLIPBOARD] = "max-clipboard", 87 [VD_AGENT_CAP_AUDIO_VOLUME_SYNC] = "audio-volume-sync", 88 [VD_AGENT_CAP_MONITORS_CONFIG_POSITION] = "monitors-config-position", 89 [VD_AGENT_CAP_FILE_XFER_DISABLED] = "file-xfer-disabled", 90 [VD_AGENT_CAP_FILE_XFER_DETAILED_ERRORS] = "file-xfer-detailed-errors", 91 #if CHECK_SPICE_PROTOCOL_VERSION(0, 14, 0) 92 [VD_AGENT_CAP_GRAPHICS_DEVICE_INFO] = "graphics-device-info", 93 #endif 94 #if CHECK_SPICE_PROTOCOL_VERSION(0, 14, 1) 95 [VD_AGENT_CAP_CLIPBOARD_NO_RELEASE_ON_REGRAB] = "clipboard-no-release-on-regrab", 96 [VD_AGENT_CAP_CLIPBOARD_GRAB_SERIAL] = "clipboard-grab-serial", 97 #endif 98 }; 99 100 static const char *msg_name[] = { 101 [VD_AGENT_MOUSE_STATE] = "mouse-state", 102 [VD_AGENT_MONITORS_CONFIG] = "monitors-config", 103 [VD_AGENT_REPLY] = "reply", 104 [VD_AGENT_CLIPBOARD] = "clipboard", 105 [VD_AGENT_DISPLAY_CONFIG] = "display-config", 106 [VD_AGENT_ANNOUNCE_CAPABILITIES] = "announce-capabilities", 107 [VD_AGENT_CLIPBOARD_GRAB] = "clipboard-grab", 108 [VD_AGENT_CLIPBOARD_REQUEST] = "clipboard-request", 109 [VD_AGENT_CLIPBOARD_RELEASE] = "clipboard-release", 110 [VD_AGENT_FILE_XFER_START] = "file-xfer-start", 111 [VD_AGENT_FILE_XFER_STATUS] = "file-xfer-status", 112 [VD_AGENT_FILE_XFER_DATA] = "file-xfer-data", 113 [VD_AGENT_CLIENT_DISCONNECTED] = "client-disconnected", 114 [VD_AGENT_MAX_CLIPBOARD] = "max-clipboard", 115 [VD_AGENT_AUDIO_VOLUME_SYNC] = "audio-volume-sync", 116 #if CHECK_SPICE_PROTOCOL_VERSION(0, 14, 0) 117 [VD_AGENT_GRAPHICS_DEVICE_INFO] = "graphics-device-info", 118 #endif 119 }; 120 121 static const char *sel_name[] = { 122 [VD_AGENT_CLIPBOARD_SELECTION_CLIPBOARD] = "clipboard", 123 [VD_AGENT_CLIPBOARD_SELECTION_PRIMARY] = "primary", 124 [VD_AGENT_CLIPBOARD_SELECTION_SECONDARY] = "secondary", 125 }; 126 127 static const char *type_name[] = { 128 [VD_AGENT_CLIPBOARD_NONE] = "none", 129 [VD_AGENT_CLIPBOARD_UTF8_TEXT] = "text", 130 [VD_AGENT_CLIPBOARD_IMAGE_PNG] = "png", 131 [VD_AGENT_CLIPBOARD_IMAGE_BMP] = "bmp", 132 [VD_AGENT_CLIPBOARD_IMAGE_TIFF] = "tiff", 133 [VD_AGENT_CLIPBOARD_IMAGE_JPG] = "jpg", 134 #if CHECK_SPICE_PROTOCOL_VERSION(0, 14, 3) 135 [VD_AGENT_CLIPBOARD_FILE_LIST] = "files", 136 #endif 137 }; 138 139 #define GET_NAME(_m, _v) \ 140 (((_v) < ARRAY_SIZE(_m) && (_m[_v])) ? (_m[_v]) : "???") 141 142 /* ------------------------------------------------------------------ */ 143 /* send messages */ 144 145 static void vdagent_send_buf(VDAgentChardev *vd) 146 { 147 uint32_t len; 148 149 while (!buffer_empty(&vd->outbuf)) { 150 len = qemu_chr_be_can_write(CHARDEV(vd)); 151 if (len == 0) { 152 return; 153 } 154 if (len > vd->outbuf.offset) { 155 len = vd->outbuf.offset; 156 } 157 qemu_chr_be_write(CHARDEV(vd), vd->outbuf.buffer, len); 158 buffer_advance(&vd->outbuf, len); 159 } 160 } 161 162 static void vdagent_send_msg(VDAgentChardev *vd, VDAgentMessage *msg) 163 { 164 uint8_t *msgbuf = (void *)msg; 165 uint32_t msgsize = sizeof(VDAgentMessage) + msg->size; 166 uint32_t msgoff = 0; 167 VDIChunkHeader chunk; 168 169 trace_vdagent_send(GET_NAME(msg_name, msg->type)); 170 171 msg->protocol = VD_AGENT_PROTOCOL; 172 173 if (vd->outbuf.offset + msgsize > VDAGENT_BUFFER_LIMIT) { 174 error_report("buffer full, dropping message"); 175 return; 176 } 177 178 while (msgoff < msgsize) { 179 chunk.port = VDP_CLIENT_PORT; 180 chunk.size = msgsize - msgoff; 181 if (chunk.size > 1024) { 182 chunk.size = 1024; 183 } 184 buffer_reserve(&vd->outbuf, sizeof(chunk) + chunk.size); 185 buffer_append(&vd->outbuf, &chunk, sizeof(chunk)); 186 buffer_append(&vd->outbuf, msgbuf + msgoff, chunk.size); 187 msgoff += chunk.size; 188 } 189 vdagent_send_buf(vd); 190 } 191 192 static void vdagent_send_caps(VDAgentChardev *vd) 193 { 194 g_autofree VDAgentMessage *msg = g_malloc0(sizeof(VDAgentMessage) + 195 sizeof(VDAgentAnnounceCapabilities) + 196 sizeof(uint32_t)); 197 VDAgentAnnounceCapabilities *caps = (void *)msg->data; 198 199 msg->type = VD_AGENT_ANNOUNCE_CAPABILITIES; 200 msg->size = sizeof(VDAgentAnnounceCapabilities) + sizeof(uint32_t); 201 if (vd->mouse) { 202 caps->caps[0] |= (1 << VD_AGENT_CAP_MOUSE_STATE); 203 } 204 if (vd->clipboard) { 205 caps->caps[0] |= (1 << VD_AGENT_CAP_CLIPBOARD_BY_DEMAND); 206 caps->caps[0] |= (1 << VD_AGENT_CAP_CLIPBOARD_SELECTION); 207 #if CHECK_SPICE_PROTOCOL_VERSION(0, 14, 1) 208 caps->caps[0] |= (1 << VD_AGENT_CAP_CLIPBOARD_GRAB_SERIAL); 209 #endif 210 } 211 212 vdagent_send_msg(vd, msg); 213 } 214 215 /* ------------------------------------------------------------------ */ 216 /* mouse events */ 217 218 static bool have_mouse(VDAgentChardev *vd) 219 { 220 return vd->mouse && 221 (vd->caps & (1 << VD_AGENT_CAP_MOUSE_STATE)); 222 } 223 224 static void vdagent_send_mouse(VDAgentChardev *vd) 225 { 226 g_autofree VDAgentMessage *msg = g_malloc0(sizeof(VDAgentMessage) + 227 sizeof(VDAgentMouseState)); 228 VDAgentMouseState *mouse = (void *)msg->data; 229 230 msg->type = VD_AGENT_MOUSE_STATE; 231 msg->size = sizeof(VDAgentMouseState); 232 233 mouse->x = vd->mouse_x; 234 mouse->y = vd->mouse_y; 235 mouse->buttons = vd->mouse_btn; 236 mouse->display_id = vd->mouse_display; 237 238 vdagent_send_msg(vd, msg); 239 } 240 241 static void vdagent_pointer_event(DeviceState *dev, QemuConsole *src, 242 InputEvent *evt) 243 { 244 static const int bmap[INPUT_BUTTON__MAX] = { 245 [INPUT_BUTTON_LEFT] = VD_AGENT_LBUTTON_MASK, 246 [INPUT_BUTTON_RIGHT] = VD_AGENT_RBUTTON_MASK, 247 [INPUT_BUTTON_MIDDLE] = VD_AGENT_MBUTTON_MASK, 248 [INPUT_BUTTON_WHEEL_UP] = VD_AGENT_UBUTTON_MASK, 249 [INPUT_BUTTON_WHEEL_DOWN] = VD_AGENT_DBUTTON_MASK, 250 #ifdef VD_AGENT_EBUTTON_MASK 251 [INPUT_BUTTON_SIDE] = VD_AGENT_SBUTTON_MASK, 252 [INPUT_BUTTON_EXTRA] = VD_AGENT_EBUTTON_MASK, 253 #endif 254 }; 255 256 VDAgentChardev *vd = container_of(dev, struct VDAgentChardev, mouse_dev); 257 InputMoveEvent *move; 258 InputBtnEvent *btn; 259 uint32_t xres, yres; 260 261 switch (evt->type) { 262 case INPUT_EVENT_KIND_ABS: 263 move = evt->u.abs.data; 264 xres = qemu_console_get_width(src, 1024); 265 yres = qemu_console_get_height(src, 768); 266 if (move->axis == INPUT_AXIS_X) { 267 vd->mouse_x = qemu_input_scale_axis(move->value, 268 INPUT_EVENT_ABS_MIN, 269 INPUT_EVENT_ABS_MAX, 270 0, xres); 271 } else if (move->axis == INPUT_AXIS_Y) { 272 vd->mouse_y = qemu_input_scale_axis(move->value, 273 INPUT_EVENT_ABS_MIN, 274 INPUT_EVENT_ABS_MAX, 275 0, yres); 276 } 277 vd->mouse_display = qemu_console_get_index(src); 278 break; 279 280 case INPUT_EVENT_KIND_BTN: 281 btn = evt->u.btn.data; 282 if (btn->down) { 283 vd->mouse_btn |= bmap[btn->button]; 284 } else { 285 vd->mouse_btn &= ~bmap[btn->button]; 286 } 287 break; 288 289 default: 290 /* keep gcc happy */ 291 break; 292 } 293 } 294 295 static void vdagent_pointer_sync(DeviceState *dev) 296 { 297 VDAgentChardev *vd = container_of(dev, struct VDAgentChardev, mouse_dev); 298 299 if (vd->caps & (1 << VD_AGENT_CAP_MOUSE_STATE)) { 300 vdagent_send_mouse(vd); 301 } 302 } 303 304 static QemuInputHandler vdagent_mouse_handler = { 305 .name = "vdagent mouse", 306 .mask = INPUT_EVENT_MASK_BTN | INPUT_EVENT_MASK_ABS, 307 .event = vdagent_pointer_event, 308 .sync = vdagent_pointer_sync, 309 }; 310 311 /* ------------------------------------------------------------------ */ 312 /* clipboard */ 313 314 static bool have_clipboard(VDAgentChardev *vd) 315 { 316 return vd->clipboard && 317 (vd->caps & (1 << VD_AGENT_CAP_CLIPBOARD_BY_DEMAND)); 318 } 319 320 static bool have_selection(VDAgentChardev *vd) 321 { 322 return vd->caps & (1 << VD_AGENT_CAP_CLIPBOARD_SELECTION); 323 } 324 325 static uint32_t type_qemu_to_vdagent(enum QemuClipboardType type) 326 { 327 switch (type) { 328 case QEMU_CLIPBOARD_TYPE_TEXT: 329 return VD_AGENT_CLIPBOARD_UTF8_TEXT; 330 default: 331 return VD_AGENT_CLIPBOARD_NONE; 332 } 333 } 334 335 static void vdagent_send_clipboard_grab(VDAgentChardev *vd, 336 QemuClipboardInfo *info) 337 { 338 g_autofree VDAgentMessage *msg = 339 g_malloc0(sizeof(VDAgentMessage) + 340 sizeof(uint32_t) * (QEMU_CLIPBOARD_TYPE__COUNT + 1) + 341 sizeof(uint32_t)); 342 uint8_t *s = msg->data; 343 uint32_t *data = (uint32_t *)msg->data; 344 uint32_t q, type; 345 346 if (have_selection(vd)) { 347 *s = info->selection; 348 data++; 349 msg->size += sizeof(uint32_t); 350 } else if (info->selection != QEMU_CLIPBOARD_SELECTION_CLIPBOARD) { 351 return; 352 } 353 354 #if CHECK_SPICE_PROTOCOL_VERSION(0, 14, 1) 355 if (vd->caps & (1 << VD_AGENT_CAP_CLIPBOARD_GRAB_SERIAL)) { 356 if (!info->has_serial) { 357 /* client should win */ 358 info->serial = vd->last_serial[info->selection]++; 359 info->has_serial = true; 360 } 361 *data = info->serial; 362 data++; 363 msg->size += sizeof(uint32_t); 364 } 365 #endif 366 367 for (q = 0; q < QEMU_CLIPBOARD_TYPE__COUNT; q++) { 368 type = type_qemu_to_vdagent(q); 369 if (type != VD_AGENT_CLIPBOARD_NONE && info->types[q].available) { 370 *data = type; 371 data++; 372 msg->size += sizeof(uint32_t); 373 } 374 } 375 376 msg->type = VD_AGENT_CLIPBOARD_GRAB; 377 vdagent_send_msg(vd, msg); 378 } 379 380 static void vdagent_send_clipboard_release(VDAgentChardev *vd, 381 QemuClipboardInfo *info) 382 { 383 g_autofree VDAgentMessage *msg = g_malloc0(sizeof(VDAgentMessage) + 384 sizeof(uint32_t)); 385 386 if (have_selection(vd)) { 387 uint8_t *s = msg->data; 388 *s = info->selection; 389 msg->size += sizeof(uint32_t); 390 } else if (info->selection != QEMU_CLIPBOARD_SELECTION_CLIPBOARD) { 391 return; 392 } 393 394 msg->type = VD_AGENT_CLIPBOARD_RELEASE; 395 vdagent_send_msg(vd, msg); 396 } 397 398 static void vdagent_send_clipboard_data(VDAgentChardev *vd, 399 QemuClipboardInfo *info, 400 QemuClipboardType type) 401 { 402 g_autofree VDAgentMessage *msg = g_malloc0(sizeof(VDAgentMessage) + 403 sizeof(uint32_t) * 2 + 404 info->types[type].size); 405 406 uint8_t *s = msg->data; 407 uint32_t *data = (uint32_t *)msg->data; 408 409 if (have_selection(vd)) { 410 *s = info->selection; 411 data++; 412 msg->size += sizeof(uint32_t); 413 } else if (info->selection != QEMU_CLIPBOARD_SELECTION_CLIPBOARD) { 414 return; 415 } 416 417 *data = type_qemu_to_vdagent(type); 418 data++; 419 msg->size += sizeof(uint32_t); 420 421 memcpy(data, info->types[type].data, info->types[type].size); 422 msg->size += info->types[type].size; 423 424 msg->type = VD_AGENT_CLIPBOARD; 425 vdagent_send_msg(vd, msg); 426 } 427 428 static void vdagent_send_empty_clipboard_data(VDAgentChardev *vd, 429 QemuClipboardSelection selection, 430 QemuClipboardType type) 431 { 432 g_autoptr(QemuClipboardInfo) info = qemu_clipboard_info_new(&vd->cbpeer, selection); 433 434 trace_vdagent_send_empty_clipboard(); 435 vdagent_send_clipboard_data(vd, info, type); 436 } 437 438 static void vdagent_clipboard_update_info(VDAgentChardev *vd, 439 QemuClipboardInfo *info) 440 { 441 QemuClipboardSelection s = info->selection; 442 QemuClipboardType type; 443 bool self_update = info->owner == &vd->cbpeer; 444 445 if (info != qemu_clipboard_info(s)) { 446 vd->cbpending[s] = 0; 447 if (!self_update) { 448 if (info->owner) { 449 vdagent_send_clipboard_grab(vd, info); 450 } else { 451 vdagent_send_clipboard_release(vd, info); 452 } 453 } 454 return; 455 } 456 457 if (self_update) { 458 return; 459 } 460 461 for (type = 0; type < QEMU_CLIPBOARD_TYPE__COUNT; type++) { 462 if (vd->cbpending[s] & (1 << type)) { 463 vd->cbpending[s] &= ~(1 << type); 464 vdagent_send_clipboard_data(vd, info, type); 465 } 466 } 467 } 468 469 static void vdagent_clipboard_reset_serial(VDAgentChardev *vd) 470 { 471 Chardev *chr = CHARDEV(vd); 472 473 /* reopen the agent connection to reset the serial state */ 474 qemu_chr_be_event(chr, CHR_EVENT_CLOSED); 475 qemu_chr_be_event(chr, CHR_EVENT_OPENED); 476 } 477 478 static void vdagent_clipboard_notify(Notifier *notifier, void *data) 479 { 480 VDAgentChardev *vd = 481 container_of(notifier, VDAgentChardev, cbpeer.notifier); 482 QemuClipboardNotify *notify = data; 483 484 switch (notify->type) { 485 case QEMU_CLIPBOARD_UPDATE_INFO: 486 vdagent_clipboard_update_info(vd, notify->info); 487 return; 488 case QEMU_CLIPBOARD_RESET_SERIAL: 489 vdagent_clipboard_reset_serial(vd); 490 return; 491 } 492 } 493 494 static void vdagent_clipboard_request(QemuClipboardInfo *info, 495 QemuClipboardType qtype) 496 { 497 VDAgentChardev *vd = container_of(info->owner, VDAgentChardev, cbpeer); 498 g_autofree VDAgentMessage *msg = g_malloc0(sizeof(VDAgentMessage) + 499 sizeof(uint32_t) * 2); 500 uint32_t type = type_qemu_to_vdagent(qtype); 501 uint8_t *s = msg->data; 502 uint32_t *data = (uint32_t *)msg->data; 503 504 if (type == VD_AGENT_CLIPBOARD_NONE) { 505 return; 506 } 507 508 if (have_selection(vd)) { 509 *s = info->selection; 510 data++; 511 msg->size += sizeof(uint32_t); 512 } 513 514 *data = type; 515 msg->size += sizeof(uint32_t); 516 517 msg->type = VD_AGENT_CLIPBOARD_REQUEST; 518 vdagent_send_msg(vd, msg); 519 } 520 521 static void vdagent_clipboard_recv_grab(VDAgentChardev *vd, uint8_t s, uint32_t size, void *data) 522 { 523 g_autoptr(QemuClipboardInfo) info = NULL; 524 525 trace_vdagent_cb_grab_selection(GET_NAME(sel_name, s)); 526 info = qemu_clipboard_info_new(&vd->cbpeer, s); 527 #if CHECK_SPICE_PROTOCOL_VERSION(0, 14, 1) 528 if (vd->caps & (1 << VD_AGENT_CAP_CLIPBOARD_GRAB_SERIAL)) { 529 if (size < sizeof(uint32_t)) { 530 /* this shouldn't happen! */ 531 return; 532 } 533 534 info->has_serial = true; 535 info->serial = *(uint32_t *)data; 536 if (info->serial < vd->last_serial[s]) { 537 /* discard lower-ordering guest grab */ 538 return; 539 } 540 vd->last_serial[s] = info->serial; 541 data += sizeof(uint32_t); 542 size -= sizeof(uint32_t); 543 } 544 #endif 545 if (size > sizeof(uint32_t) * 10) { 546 /* 547 * spice has 6 types as of 2021. Limiting to 10 entries 548 * so we we have some wiggle room. 549 */ 550 return; 551 } 552 while (size >= sizeof(uint32_t)) { 553 trace_vdagent_cb_grab_type(GET_NAME(type_name, *(uint32_t *)data)); 554 switch (*(uint32_t *)data) { 555 case VD_AGENT_CLIPBOARD_UTF8_TEXT: 556 info->types[QEMU_CLIPBOARD_TYPE_TEXT].available = true; 557 break; 558 default: 559 break; 560 } 561 data += sizeof(uint32_t); 562 size -= sizeof(uint32_t); 563 } 564 qemu_clipboard_update(info); 565 } 566 567 static void vdagent_clipboard_recv_request(VDAgentChardev *vd, uint8_t s, uint32_t size, void *data) 568 { 569 QemuClipboardType type; 570 QemuClipboardInfo *info; 571 572 if (size < sizeof(uint32_t)) { 573 return; 574 } 575 switch (*(uint32_t *)data) { 576 case VD_AGENT_CLIPBOARD_UTF8_TEXT: 577 type = QEMU_CLIPBOARD_TYPE_TEXT; 578 break; 579 default: 580 return; 581 } 582 583 info = qemu_clipboard_info(s); 584 if (info && info->types[type].available && info->owner != &vd->cbpeer) { 585 if (info->types[type].data) { 586 vdagent_send_clipboard_data(vd, info, type); 587 } else { 588 vd->cbpending[s] |= (1 << type); 589 qemu_clipboard_request(info, type); 590 } 591 } else { 592 vdagent_send_empty_clipboard_data(vd, s, type); 593 } 594 } 595 596 static void vdagent_clipboard_recv_data(VDAgentChardev *vd, uint8_t s, uint32_t size, void *data) 597 { 598 QemuClipboardType type; 599 600 if (size < sizeof(uint32_t)) { 601 return; 602 } 603 switch (*(uint32_t *)data) { 604 case VD_AGENT_CLIPBOARD_UTF8_TEXT: 605 type = QEMU_CLIPBOARD_TYPE_TEXT; 606 break; 607 default: 608 return; 609 } 610 data += 4; 611 size -= 4; 612 613 if (qemu_clipboard_peer_owns(&vd->cbpeer, s)) { 614 qemu_clipboard_set_data(&vd->cbpeer, qemu_clipboard_info(s), 615 type, size, data, true); 616 } 617 } 618 619 static void vdagent_clipboard_recv_release(VDAgentChardev *vd, uint8_t s) 620 { 621 qemu_clipboard_peer_release(&vd->cbpeer, s); 622 } 623 624 static void vdagent_chr_recv_clipboard(VDAgentChardev *vd, VDAgentMessage *msg) 625 { 626 uint8_t s = VD_AGENT_CLIPBOARD_SELECTION_CLIPBOARD; 627 uint32_t size = msg->size; 628 void *data = msg->data; 629 630 if (have_selection(vd)) { 631 if (size < 4) { 632 return; 633 } 634 s = *(uint8_t *)data; 635 if (s >= QEMU_CLIPBOARD_SELECTION__COUNT) { 636 return; 637 } 638 data += 4; 639 size -= 4; 640 } 641 642 switch (msg->type) { 643 case VD_AGENT_CLIPBOARD_GRAB: 644 return vdagent_clipboard_recv_grab(vd, s, size, data); 645 case VD_AGENT_CLIPBOARD_REQUEST: 646 return vdagent_clipboard_recv_request(vd, s, size, data); 647 case VD_AGENT_CLIPBOARD: /* data */ 648 return vdagent_clipboard_recv_data(vd, s, size, data); 649 case VD_AGENT_CLIPBOARD_RELEASE: 650 return vdagent_clipboard_recv_release(vd, s); 651 default: 652 g_assert_not_reached(); 653 } 654 } 655 656 /* ------------------------------------------------------------------ */ 657 /* chardev backend */ 658 659 static void vdagent_chr_open(Chardev *chr, 660 ChardevBackend *backend, 661 bool *be_opened, 662 Error **errp) 663 { 664 VDAgentChardev *vd = QEMU_VDAGENT_CHARDEV(chr); 665 ChardevQemuVDAgent *cfg = backend->u.qemu_vdagent.data; 666 667 #if HOST_BIG_ENDIAN 668 /* 669 * TODO: vdagent protocol is defined to be LE, 670 * so we have to byteswap everything on BE hosts. 671 */ 672 error_setg(errp, "vdagent is not supported on bigendian hosts"); 673 return; 674 #endif 675 676 if (migrate_add_blocker(vd->migration_blocker, errp) != 0) { 677 return; 678 } 679 680 vd->mouse = VDAGENT_MOUSE_DEFAULT; 681 if (cfg->has_mouse) { 682 vd->mouse = cfg->mouse; 683 } 684 685 vd->clipboard = VDAGENT_CLIPBOARD_DEFAULT; 686 if (cfg->has_clipboard) { 687 vd->clipboard = cfg->clipboard; 688 } 689 690 if (vd->mouse) { 691 vd->mouse_hs = qemu_input_handler_register(&vd->mouse_dev, 692 &vdagent_mouse_handler); 693 } 694 695 *be_opened = true; 696 } 697 698 static void vdagent_chr_recv_caps(VDAgentChardev *vd, VDAgentMessage *msg) 699 { 700 VDAgentAnnounceCapabilities *caps = (void *)msg->data; 701 int i; 702 703 if (msg->size < (sizeof(VDAgentAnnounceCapabilities) + 704 sizeof(uint32_t))) { 705 return; 706 } 707 708 for (i = 0; i < ARRAY_SIZE(cap_name); i++) { 709 if (caps->caps[0] & (1 << i)) { 710 trace_vdagent_peer_cap(GET_NAME(cap_name, i)); 711 } 712 } 713 714 vd->caps = caps->caps[0]; 715 if (caps->request) { 716 vdagent_send_caps(vd); 717 } 718 if (have_mouse(vd) && vd->mouse_hs) { 719 qemu_input_handler_activate(vd->mouse_hs); 720 } 721 if (have_clipboard(vd) && vd->cbpeer.notifier.notify == NULL) { 722 memset(vd->last_serial, 0, sizeof(vd->last_serial)); 723 vd->cbpeer.name = "vdagent"; 724 vd->cbpeer.notifier.notify = vdagent_clipboard_notify; 725 vd->cbpeer.request = vdagent_clipboard_request; 726 qemu_clipboard_peer_register(&vd->cbpeer); 727 } 728 } 729 730 static void vdagent_chr_recv_msg(VDAgentChardev *vd, VDAgentMessage *msg) 731 { 732 trace_vdagent_recv_msg(GET_NAME(msg_name, msg->type), msg->size); 733 734 switch (msg->type) { 735 case VD_AGENT_ANNOUNCE_CAPABILITIES: 736 vdagent_chr_recv_caps(vd, msg); 737 break; 738 case VD_AGENT_CLIPBOARD: 739 case VD_AGENT_CLIPBOARD_GRAB: 740 case VD_AGENT_CLIPBOARD_REQUEST: 741 case VD_AGENT_CLIPBOARD_RELEASE: 742 if (have_clipboard(vd)) { 743 vdagent_chr_recv_clipboard(vd, msg); 744 } 745 break; 746 default: 747 break; 748 } 749 } 750 751 static void vdagent_reset_xbuf(VDAgentChardev *vd) 752 { 753 g_clear_pointer(&vd->xbuf, g_free); 754 vd->xoff = 0; 755 vd->xsize = 0; 756 } 757 758 static void vdagent_chr_recv_chunk(VDAgentChardev *vd) 759 { 760 VDAgentMessage *msg = (void *)vd->msgbuf; 761 762 if (!vd->xsize) { 763 if (vd->msgsize < sizeof(*msg)) { 764 error_report("%s: message too small: %d < %zd", __func__, 765 vd->msgsize, sizeof(*msg)); 766 return; 767 } 768 if (vd->msgsize == msg->size + sizeof(*msg)) { 769 vdagent_chr_recv_msg(vd, msg); 770 return; 771 } 772 } 773 774 if (!vd->xsize) { 775 vd->xsize = msg->size + sizeof(*msg); 776 vd->xbuf = g_malloc0(vd->xsize); 777 } 778 779 if (vd->xoff + vd->msgsize > vd->xsize) { 780 error_report("%s: Oops: %d+%d > %d", __func__, 781 vd->xoff, vd->msgsize, vd->xsize); 782 vdagent_reset_xbuf(vd); 783 return; 784 } 785 786 memcpy(vd->xbuf + vd->xoff, vd->msgbuf, vd->msgsize); 787 vd->xoff += vd->msgsize; 788 if (vd->xoff < vd->xsize) { 789 return; 790 } 791 792 msg = (void *)vd->xbuf; 793 vdagent_chr_recv_msg(vd, msg); 794 vdagent_reset_xbuf(vd); 795 } 796 797 static void vdagent_reset_bufs(VDAgentChardev *vd) 798 { 799 memset(&vd->chunk, 0, sizeof(vd->chunk)); 800 vd->chunksize = 0; 801 g_free(vd->msgbuf); 802 vd->msgbuf = NULL; 803 vd->msgsize = 0; 804 } 805 806 static int vdagent_chr_write(Chardev *chr, const uint8_t *buf, int len) 807 { 808 VDAgentChardev *vd = QEMU_VDAGENT_CHARDEV(chr); 809 uint32_t copy, ret = len; 810 811 while (len) { 812 if (vd->chunksize < sizeof(vd->chunk)) { 813 copy = sizeof(vd->chunk) - vd->chunksize; 814 if (copy > len) { 815 copy = len; 816 } 817 memcpy((void *)(&vd->chunk) + vd->chunksize, buf, copy); 818 vd->chunksize += copy; 819 buf += copy; 820 len -= copy; 821 if (vd->chunksize < sizeof(vd->chunk)) { 822 break; 823 } 824 825 assert(vd->msgbuf == NULL); 826 vd->msgbuf = g_malloc0(vd->chunk.size); 827 } 828 829 copy = vd->chunk.size - vd->msgsize; 830 if (copy > len) { 831 copy = len; 832 } 833 memcpy(vd->msgbuf + vd->msgsize, buf, copy); 834 vd->msgsize += copy; 835 buf += copy; 836 len -= copy; 837 838 if (vd->msgsize == vd->chunk.size) { 839 trace_vdagent_recv_chunk(vd->chunk.size); 840 vdagent_chr_recv_chunk(vd); 841 vdagent_reset_bufs(vd); 842 } 843 } 844 845 return ret; 846 } 847 848 static void vdagent_chr_accept_input(Chardev *chr) 849 { 850 VDAgentChardev *vd = QEMU_VDAGENT_CHARDEV(chr); 851 852 vdagent_send_buf(vd); 853 } 854 855 static void vdagent_disconnect(VDAgentChardev *vd) 856 { 857 buffer_reset(&vd->outbuf); 858 vdagent_reset_bufs(vd); 859 vd->caps = 0; 860 if (vd->mouse_hs) { 861 qemu_input_handler_deactivate(vd->mouse_hs); 862 } 863 if (vd->cbpeer.notifier.notify) { 864 qemu_clipboard_peer_unregister(&vd->cbpeer); 865 memset(&vd->cbpeer, 0, sizeof(vd->cbpeer)); 866 } 867 } 868 869 static void vdagent_chr_set_fe_open(struct Chardev *chr, int fe_open) 870 { 871 if (!fe_open) { 872 trace_vdagent_close(); 873 return; 874 } 875 876 trace_vdagent_open(); 877 } 878 879 static void vdagent_chr_parse(QemuOpts *opts, ChardevBackend *backend, 880 Error **errp) 881 { 882 ChardevQemuVDAgent *cfg; 883 884 backend->type = CHARDEV_BACKEND_KIND_QEMU_VDAGENT; 885 cfg = backend->u.qemu_vdagent.data = g_new0(ChardevQemuVDAgent, 1); 886 qemu_chr_parse_common(opts, qapi_ChardevQemuVDAgent_base(cfg)); 887 cfg->has_mouse = true; 888 cfg->mouse = qemu_opt_get_bool(opts, "mouse", VDAGENT_MOUSE_DEFAULT); 889 cfg->has_clipboard = true; 890 cfg->clipboard = qemu_opt_get_bool(opts, "clipboard", VDAGENT_CLIPBOARD_DEFAULT); 891 } 892 893 /* ------------------------------------------------------------------ */ 894 895 static void vdagent_chr_class_init(ObjectClass *oc, void *data) 896 { 897 ChardevClass *cc = CHARDEV_CLASS(oc); 898 899 cc->parse = vdagent_chr_parse; 900 cc->open = vdagent_chr_open; 901 cc->chr_write = vdagent_chr_write; 902 cc->chr_set_fe_open = vdagent_chr_set_fe_open; 903 cc->chr_accept_input = vdagent_chr_accept_input; 904 } 905 906 static void vdagent_chr_init(Object *obj) 907 { 908 VDAgentChardev *vd = QEMU_VDAGENT_CHARDEV(obj); 909 910 buffer_init(&vd->outbuf, "vdagent-outbuf"); 911 error_setg(&vd->migration_blocker, 912 "The vdagent chardev doesn't yet support migration"); 913 } 914 915 static void vdagent_chr_fini(Object *obj) 916 { 917 VDAgentChardev *vd = QEMU_VDAGENT_CHARDEV(obj); 918 919 migrate_del_blocker(vd->migration_blocker); 920 vdagent_disconnect(vd); 921 buffer_free(&vd->outbuf); 922 error_free(vd->migration_blocker); 923 } 924 925 static const TypeInfo vdagent_chr_type_info = { 926 .name = TYPE_CHARDEV_QEMU_VDAGENT, 927 .parent = TYPE_CHARDEV, 928 .instance_size = sizeof(VDAgentChardev), 929 .instance_init = vdagent_chr_init, 930 .instance_finalize = vdagent_chr_fini, 931 .class_init = vdagent_chr_class_init, 932 }; 933 934 static void register_types(void) 935 { 936 type_register_static(&vdagent_chr_type_info); 937 } 938 939 type_init(register_types); 940