1 /*- 2 * Copyright (c) 2018 VMware, Inc. All Rights Reserved. 3 * 4 * SPDX-License-Identifier: (BSD-2-Clause OR GPL-2.0) 5 */ 6 7 /* VMCI QueuePair API implementation. */ 8 9 #include <sys/cdefs.h> 10 __FBSDID("$FreeBSD$"); 11 12 #include "vmci.h" 13 #include "vmci_driver.h" 14 #include "vmci_event.h" 15 #include "vmci_kernel_api.h" 16 #include "vmci_kernel_defs.h" 17 #include "vmci_queue_pair.h" 18 19 #define LGPFX "vmci_queue_pair: " 20 21 struct queue_pair_entry { 22 vmci_list_item(queue_pair_entry) list_item; 23 struct vmci_handle handle; 24 vmci_id peer; 25 uint32_t flags; 26 uint64_t produce_size; 27 uint64_t consume_size; 28 uint32_t ref_count; 29 }; 30 31 struct qp_guest_endpoint { 32 struct queue_pair_entry qp; 33 uint64_t num_ppns; 34 void *produce_q; 35 void *consume_q; 36 bool hibernate_failure; 37 struct ppn_set ppn_set; 38 }; 39 40 struct queue_pair_list { 41 vmci_list(queue_pair_entry) head; 42 volatile int hibernate; 43 vmci_mutex mutex; 44 }; 45 46 #define QPE_NUM_PAGES(_QPE) \ 47 ((uint32_t)(CEILING(_QPE.produce_size, PAGE_SIZE) + \ 48 CEILING(_QPE.consume_size, PAGE_SIZE) + 2)) 49 50 static struct queue_pair_list qp_guest_endpoints; 51 52 static struct queue_pair_entry *queue_pair_list_find_entry( 53 struct queue_pair_list *qp_list, struct vmci_handle handle); 54 static void queue_pair_list_add_entry(struct queue_pair_list *qp_list, 55 struct queue_pair_entry *entry); 56 static void queue_pair_list_remove_entry(struct queue_pair_list *qp_list, 57 struct queue_pair_entry *entry); 58 static struct queue_pair_entry *queue_pair_list_get_head( 59 struct queue_pair_list *qp_list); 60 static int queue_pair_notify_peer_local(bool attach, 61 struct vmci_handle handle); 62 static struct qp_guest_endpoint *qp_guest_endpoint_create( 63 struct vmci_handle handle, vmci_id peer, uint32_t flags, 64 uint64_t produce_size, uint64_t consume_size, 65 void *produce_q, void *consume_q); 66 static void qp_guest_endpoint_destroy(struct qp_guest_endpoint *entry); 67 static int vmci_queue_pair_alloc_hypercall( 68 const struct qp_guest_endpoint *entry); 69 static int vmci_queue_pair_alloc_guest_work(struct vmci_handle *handle, 70 struct vmci_queue **produce_q, uint64_t produce_size, 71 struct vmci_queue **consume_q, uint64_t consume_size, 72 vmci_id peer, uint32_t flags, 73 vmci_privilege_flags priv_flags); 74 static int vmci_queue_pair_detach_guest_work(struct vmci_handle handle); 75 static int vmci_queue_pair_detach_hypercall(struct vmci_handle handle); 76 77 /* 78 *------------------------------------------------------------------------------ 79 * 80 * vmci_queue_pair_alloc -- 81 * 82 * Allocates a VMCI QueuePair. Only checks validity of input arguments. The 83 * real work is done in the host or guest specific function. 84 * 85 * Results: 86 * VMCI_SUCCESS on success, appropriate error code otherwise. 87 * 88 * Side effects: 89 * None. 90 * 91 *------------------------------------------------------------------------------ 92 */ 93 94 int 95 vmci_queue_pair_alloc(struct vmci_handle *handle, struct vmci_queue **produce_q, 96 uint64_t produce_size, struct vmci_queue **consume_q, uint64_t consume_size, 97 vmci_id peer, uint32_t flags, vmci_privilege_flags priv_flags) 98 { 99 100 if (!handle || !produce_q || !consume_q || 101 (!produce_size && !consume_size) || (flags & ~VMCI_QP_ALL_FLAGS)) 102 return (VMCI_ERROR_INVALID_ARGS); 103 104 return (vmci_queue_pair_alloc_guest_work(handle, produce_q, 105 produce_size, consume_q, consume_size, peer, flags, priv_flags)); 106 } 107 108 /* 109 *------------------------------------------------------------------------------ 110 * 111 * vmci_queue_pair_detach -- 112 * 113 * Detaches from a VMCI QueuePair. Only checks validity of input argument. 114 * Real work is done in the host or guest specific function. 115 * 116 * Results: 117 * Success or failure. 118 * 119 * Side effects: 120 * Memory is freed. 121 * 122 *------------------------------------------------------------------------------ 123 */ 124 125 int 126 vmci_queue_pair_detach(struct vmci_handle handle) 127 { 128 129 if (VMCI_HANDLE_INVALID(handle)) 130 return (VMCI_ERROR_INVALID_ARGS); 131 132 return (vmci_queue_pair_detach_guest_work(handle)); 133 } 134 135 /* 136 *------------------------------------------------------------------------------ 137 * 138 * queue_pair_list_init -- 139 * 140 * Initializes the list of QueuePairs. 141 * 142 * Results: 143 * Success or failure. 144 * 145 * Side effects: 146 * None. 147 * 148 *------------------------------------------------------------------------------ 149 */ 150 151 static inline int 152 queue_pair_list_init(struct queue_pair_list *qp_list) 153 { 154 int ret; 155 156 vmci_list_init(&qp_list->head); 157 atomic_store_int(&qp_list->hibernate, 0); 158 ret = vmci_mutex_init(&qp_list->mutex, "VMCI QP List lock"); 159 return (ret); 160 } 161 162 /* 163 *------------------------------------------------------------------------------ 164 * 165 * queue_pair_list_destroy -- 166 * 167 * Destroy the list's mutex. 168 * 169 * Results: 170 * None. 171 * 172 * Side effects: 173 * None. 174 * 175 *------------------------------------------------------------------------------ 176 */ 177 178 static inline void 179 queue_pair_list_destroy(struct queue_pair_list *qp_list) 180 { 181 182 vmci_mutex_destroy(&qp_list->mutex); 183 vmci_list_init(&qp_list->head); 184 } 185 186 /* 187 *------------------------------------------------------------------------------ 188 * 189 * queue_pair_list_find_entry -- 190 * 191 * Finds the entry in the list corresponding to a given handle. Assumes that 192 * the list is locked. 193 * 194 * Results: 195 * Pointer to entry. 196 * 197 * Side effects: 198 * None. 199 * 200 *------------------------------------------------------------------------------ 201 */ 202 203 static struct queue_pair_entry * 204 queue_pair_list_find_entry(struct queue_pair_list *qp_list, 205 struct vmci_handle handle) 206 { 207 struct queue_pair_entry *next; 208 209 if (VMCI_HANDLE_INVALID(handle)) 210 return (NULL); 211 212 vmci_list_scan(next, &qp_list->head, list_item) { 213 if (VMCI_HANDLE_EQUAL(next->handle, handle)) 214 return (next); 215 } 216 217 return (NULL); 218 } 219 220 /* 221 *------------------------------------------------------------------------------ 222 * 223 * queue_pair_list_add_entry -- 224 * 225 * Adds the given entry to the list. Assumes that the list is locked. 226 * 227 * Results: 228 * None. 229 * 230 * Side effects: 231 * None. 232 * 233 *------------------------------------------------------------------------------ 234 */ 235 236 static void 237 queue_pair_list_add_entry(struct queue_pair_list *qp_list, 238 struct queue_pair_entry *entry) 239 { 240 241 if (entry) 242 vmci_list_insert(&qp_list->head, entry, list_item); 243 } 244 245 /* 246 *------------------------------------------------------------------------------ 247 * 248 * queue_pair_list_remove_entry -- 249 * 250 * Removes the given entry from the list. Assumes that the list is locked. 251 * 252 * Results: 253 * None. 254 * 255 * Side effects: 256 * None. 257 * 258 *------------------------------------------------------------------------------ 259 */ 260 261 static void 262 queue_pair_list_remove_entry(struct queue_pair_list *qp_list, 263 struct queue_pair_entry *entry) 264 { 265 266 if (entry) 267 vmci_list_remove(entry, list_item); 268 } 269 270 /* 271 *------------------------------------------------------------------------------ 272 * 273 * queue_pair_list_get_head -- 274 * 275 * Returns the entry from the head of the list. Assumes that the list is 276 * locked. 277 * 278 * Results: 279 * Pointer to entry. 280 * 281 * Side effects: 282 * None. 283 * 284 *------------------------------------------------------------------------------ 285 */ 286 287 static struct queue_pair_entry * 288 queue_pair_list_get_head(struct queue_pair_list *qp_list) 289 { 290 291 return (vmci_list_first(&qp_list->head)); 292 } 293 294 /* 295 *------------------------------------------------------------------------------ 296 * 297 * vmci_qp_guest_endpoints_init -- 298 * 299 * Initalizes data structure state keeping track of queue pair guest 300 * endpoints. 301 * 302 * Results: 303 * VMCI_SUCCESS on success and appropriate failure code otherwise. 304 * 305 * Side effects: 306 * None. 307 * 308 *------------------------------------------------------------------------------ 309 */ 310 311 int 312 vmci_qp_guest_endpoints_init(void) 313 { 314 315 return (queue_pair_list_init(&qp_guest_endpoints)); 316 } 317 318 /* 319 *------------------------------------------------------------------------------ 320 * 321 * vmci_qp_guest_endpoints_exit -- 322 * 323 * Destroys all guest queue pair endpoints. If active guest queue pairs 324 * still exist, hypercalls to attempt detach from these queue pairs will be 325 * made. Any failure to detach is silently ignored. 326 * 327 * Results: 328 * None. 329 * 330 * Side effects: 331 * None. 332 * 333 *------------------------------------------------------------------------------ 334 */ 335 336 void 337 vmci_qp_guest_endpoints_exit(void) 338 { 339 struct qp_guest_endpoint *entry; 340 341 vmci_mutex_acquire(&qp_guest_endpoints.mutex); 342 343 while ((entry = 344 (struct qp_guest_endpoint *)queue_pair_list_get_head( 345 &qp_guest_endpoints)) != NULL) { 346 /* 347 * Don't make a hypercall for local QueuePairs. 348 */ 349 if (!(entry->qp.flags & VMCI_QPFLAG_LOCAL)) 350 vmci_queue_pair_detach_hypercall(entry->qp.handle); 351 /* 352 * We cannot fail the exit, so let's reset ref_count. 353 */ 354 entry->qp.ref_count = 0; 355 queue_pair_list_remove_entry(&qp_guest_endpoints, &entry->qp); 356 qp_guest_endpoint_destroy(entry); 357 } 358 359 atomic_store_int(&qp_guest_endpoints.hibernate, 0); 360 vmci_mutex_release(&qp_guest_endpoints.mutex); 361 queue_pair_list_destroy(&qp_guest_endpoints); 362 } 363 364 /* 365 *------------------------------------------------------------------------------ 366 * 367 * vmci_qp_guest_endpoints_sync -- 368 * 369 * Use this as a synchronization point when setting globals, for example, 370 * during device shutdown. 371 * 372 * Results: 373 * true. 374 * 375 * Side effects: 376 * None. 377 * 378 *------------------------------------------------------------------------------ 379 */ 380 381 void 382 vmci_qp_guest_endpoints_sync(void) 383 { 384 385 vmci_mutex_acquire(&qp_guest_endpoints.mutex); 386 vmci_mutex_release(&qp_guest_endpoints.mutex); 387 } 388 389 /* 390 *------------------------------------------------------------------------------ 391 * 392 * qp_guest_endpoint_create -- 393 * 394 * Allocates and initializes a qp_guest_endpoint structure. Allocates a 395 * QueuePair rid (and handle) iff the given entry has an invalid handle. 396 * 0 through VMCI_RESERVED_RESOURCE_ID_MAX are reserved handles. Assumes 397 * that the QP list mutex is held by the caller. 398 * 399 * Results: 400 * Pointer to structure intialized. 401 * 402 * Side effects: 403 * None. 404 * 405 *------------------------------------------------------------------------------ 406 */ 407 408 struct qp_guest_endpoint * 409 qp_guest_endpoint_create(struct vmci_handle handle, vmci_id peer, 410 uint32_t flags, uint64_t produce_size, uint64_t consume_size, 411 void *produce_q, void *consume_q) 412 { 413 struct qp_guest_endpoint *entry; 414 static vmci_id queue_pair_rid; 415 const uint64_t num_ppns = CEILING(produce_size, PAGE_SIZE) + 416 CEILING(consume_size, PAGE_SIZE) + 417 2; /* One page each for the queue headers. */ 418 419 queue_pair_rid = VMCI_RESERVED_RESOURCE_ID_MAX + 1; 420 421 ASSERT((produce_size || consume_size) && produce_q && consume_q); 422 423 if (VMCI_HANDLE_INVALID(handle)) { 424 vmci_id context_id = vmci_get_context_id(); 425 vmci_id old_rid = queue_pair_rid; 426 427 /* 428 * Generate a unique QueuePair rid. Keep on trying until we 429 * wrap around in the RID space. 430 */ 431 ASSERT(old_rid > VMCI_RESERVED_RESOURCE_ID_MAX); 432 do { 433 handle = VMCI_MAKE_HANDLE(context_id, queue_pair_rid); 434 entry = 435 (struct qp_guest_endpoint *) 436 queue_pair_list_find_entry(&qp_guest_endpoints, 437 handle); 438 queue_pair_rid++; 439 if (UNLIKELY(!queue_pair_rid)) { 440 /* 441 * Skip the reserved rids. 442 */ 443 queue_pair_rid = 444 VMCI_RESERVED_RESOURCE_ID_MAX + 1; 445 } 446 } while (entry && queue_pair_rid != old_rid); 447 448 if (UNLIKELY(entry != NULL)) { 449 ASSERT(queue_pair_rid == old_rid); 450 /* 451 * We wrapped around --- no rids were free. 452 */ 453 return (NULL); 454 } 455 } 456 457 ASSERT(!VMCI_HANDLE_INVALID(handle) && 458 queue_pair_list_find_entry(&qp_guest_endpoints, handle) == NULL); 459 entry = vmci_alloc_kernel_mem(sizeof(*entry), VMCI_MEMORY_NORMAL); 460 if (entry) { 461 entry->qp.handle = handle; 462 entry->qp.peer = peer; 463 entry->qp.flags = flags; 464 entry->qp.produce_size = produce_size; 465 entry->qp.consume_size = consume_size; 466 entry->qp.ref_count = 0; 467 entry->num_ppns = num_ppns; 468 memset(&entry->ppn_set, 0, sizeof(entry->ppn_set)); 469 entry->produce_q = produce_q; 470 entry->consume_q = consume_q; 471 } 472 return (entry); 473 } 474 475 /* 476 *------------------------------------------------------------------------------ 477 * 478 * qp_guest_endpoint_destroy -- 479 * 480 * Frees a qp_guest_endpoint structure. 481 * 482 * Results: 483 * None. 484 * 485 * Side effects: 486 * None. 487 * 488 *------------------------------------------------------------------------------ 489 */ 490 491 void 492 qp_guest_endpoint_destroy(struct qp_guest_endpoint *entry) 493 { 494 495 ASSERT(entry); 496 ASSERT(entry->qp.ref_count == 0); 497 498 vmci_free_ppn_set(&entry->ppn_set); 499 vmci_free_queue(entry->produce_q, entry->qp.produce_size); 500 vmci_free_queue(entry->consume_q, entry->qp.consume_size); 501 vmci_free_kernel_mem(entry, sizeof(*entry)); 502 } 503 504 /* 505 *------------------------------------------------------------------------------ 506 * 507 * vmci_queue_pair_alloc_hypercall -- 508 * 509 * Helper to make a QueuePairAlloc hypercall when the driver is 510 * supporting a guest device. 511 * 512 * Results: 513 * Result of the hypercall. 514 * 515 * Side effects: 516 * Memory is allocated & freed. 517 * 518 *------------------------------------------------------------------------------ 519 */ 520 static int 521 vmci_queue_pair_alloc_hypercall(const struct qp_guest_endpoint *entry) 522 { 523 struct vmci_queue_pair_alloc_msg *alloc_msg; 524 size_t msg_size; 525 int result; 526 527 if (!entry || entry->num_ppns <= 2) 528 return (VMCI_ERROR_INVALID_ARGS); 529 530 ASSERT(!(entry->qp.flags & VMCI_QPFLAG_LOCAL)); 531 532 msg_size = sizeof(*alloc_msg) + (size_t)entry->num_ppns * sizeof(PPN); 533 alloc_msg = vmci_alloc_kernel_mem(msg_size, VMCI_MEMORY_NORMAL); 534 if (!alloc_msg) 535 return (VMCI_ERROR_NO_MEM); 536 537 alloc_msg->hdr.dst = VMCI_MAKE_HANDLE(VMCI_HYPERVISOR_CONTEXT_ID, 538 VMCI_QUEUEPAIR_ALLOC); 539 alloc_msg->hdr.src = VMCI_ANON_SRC_HANDLE; 540 alloc_msg->hdr.payload_size = msg_size - VMCI_DG_HEADERSIZE; 541 alloc_msg->handle = entry->qp.handle; 542 alloc_msg->peer = entry->qp.peer; 543 alloc_msg->flags = entry->qp.flags; 544 alloc_msg->produce_size = entry->qp.produce_size; 545 alloc_msg->consume_size = entry->qp.consume_size; 546 alloc_msg->num_ppns = entry->num_ppns; 547 result = vmci_populate_ppn_list((uint8_t *)alloc_msg + 548 sizeof(*alloc_msg), &entry->ppn_set); 549 if (result == VMCI_SUCCESS) 550 result = vmci_send_datagram((struct vmci_datagram *)alloc_msg); 551 vmci_free_kernel_mem(alloc_msg, msg_size); 552 553 return (result); 554 } 555 556 /* 557 *------------------------------------------------------------------------------ 558 * 559 * vmci_queue_pair_alloc_guest_work -- 560 * 561 * This functions handles the actual allocation of a VMCI queue pair guest 562 * endpoint. Allocates physical pages for the queue pair. It makes OS 563 * dependent calls through generic wrappers. 564 * 565 * Results: 566 * Success or failure. 567 * 568 * Side effects: 569 * Memory is allocated. 570 * 571 *------------------------------------------------------------------------------ 572 */ 573 574 static int 575 vmci_queue_pair_alloc_guest_work(struct vmci_handle *handle, 576 struct vmci_queue **produce_q, uint64_t produce_size, 577 struct vmci_queue **consume_q, uint64_t consume_size, vmci_id peer, 578 uint32_t flags, vmci_privilege_flags priv_flags) 579 { 580 struct qp_guest_endpoint *queue_pair_entry = NULL; 581 void *my_consume_q = NULL; 582 void *my_produce_q = NULL; 583 const uint64_t num_consume_pages = CEILING(consume_size, PAGE_SIZE) + 1; 584 const uint64_t num_produce_pages = CEILING(produce_size, PAGE_SIZE) + 1; 585 int result; 586 587 ASSERT(handle && produce_q && consume_q && 588 (produce_size || consume_size)); 589 590 if (priv_flags != VMCI_NO_PRIVILEGE_FLAGS) 591 return (VMCI_ERROR_NO_ACCESS); 592 593 vmci_mutex_acquire(&qp_guest_endpoints.mutex); 594 595 if ((atomic_load_int(&qp_guest_endpoints.hibernate) == 1) && 596 !(flags & VMCI_QPFLAG_LOCAL)) { 597 /* 598 * While guest OS is in hibernate state, creating non-local 599 * queue pairs is not allowed after the point where the VMCI 600 * guest driver converted the existing queue pairs to local 601 * ones. 602 */ 603 604 result = VMCI_ERROR_UNAVAILABLE; 605 goto error; 606 } 607 608 if ((queue_pair_entry = 609 (struct qp_guest_endpoint *)queue_pair_list_find_entry( 610 &qp_guest_endpoints, *handle)) != NULL) { 611 if (queue_pair_entry->qp.flags & VMCI_QPFLAG_LOCAL) { 612 /* Local attach case. */ 613 if (queue_pair_entry->qp.ref_count > 1) { 614 VMCI_LOG_DEBUG(LGPFX"Error attempting to " 615 "attach more than once.\n"); 616 result = VMCI_ERROR_UNAVAILABLE; 617 goto error_keep_entry; 618 } 619 620 if (queue_pair_entry->qp.produce_size != consume_size || 621 queue_pair_entry->qp.consume_size != produce_size || 622 queue_pair_entry->qp.flags != 623 (flags & ~VMCI_QPFLAG_ATTACH_ONLY)) { 624 VMCI_LOG_DEBUG(LGPFX"Error mismatched " 625 "queue pair in local attach.\n"); 626 result = VMCI_ERROR_QUEUEPAIR_MISMATCH; 627 goto error_keep_entry; 628 } 629 630 /* 631 * Do a local attach. We swap the consume and produce 632 * queues for the attacher and deliver an attach event. 633 */ 634 result = queue_pair_notify_peer_local(true, *handle); 635 if (result < VMCI_SUCCESS) 636 goto error_keep_entry; 637 my_produce_q = queue_pair_entry->consume_q; 638 my_consume_q = queue_pair_entry->produce_q; 639 goto out; 640 } 641 result = VMCI_ERROR_ALREADY_EXISTS; 642 goto error_keep_entry; 643 } 644 645 my_produce_q = vmci_alloc_queue(produce_size, flags); 646 if (!my_produce_q) { 647 VMCI_LOG_WARNING(LGPFX"Error allocating pages for produce " 648 "queue.\n"); 649 result = VMCI_ERROR_NO_MEM; 650 goto error; 651 } 652 653 my_consume_q = vmci_alloc_queue(consume_size, flags); 654 if (!my_consume_q) { 655 VMCI_LOG_WARNING(LGPFX"Error allocating pages for consume " 656 "queue.\n"); 657 result = VMCI_ERROR_NO_MEM; 658 goto error; 659 } 660 661 queue_pair_entry = qp_guest_endpoint_create(*handle, peer, flags, 662 produce_size, consume_size, my_produce_q, my_consume_q); 663 if (!queue_pair_entry) { 664 VMCI_LOG_WARNING(LGPFX"Error allocating memory in %s.\n", 665 __FUNCTION__); 666 result = VMCI_ERROR_NO_MEM; 667 goto error; 668 } 669 670 result = vmci_alloc_ppn_set(my_produce_q, num_produce_pages, 671 my_consume_q, num_consume_pages, &queue_pair_entry->ppn_set); 672 if (result < VMCI_SUCCESS) { 673 VMCI_LOG_WARNING(LGPFX"vmci_alloc_ppn_set failed.\n"); 674 goto error; 675 } 676 677 /* 678 * It's only necessary to notify the host if this queue pair will be 679 * attached to from another context. 680 */ 681 if (queue_pair_entry->qp.flags & VMCI_QPFLAG_LOCAL) { 682 /* Local create case. */ 683 vmci_id context_id = vmci_get_context_id(); 684 685 /* 686 * Enforce similar checks on local queue pairs as we do for 687 * regular ones. The handle's context must match the creator 688 * or attacher context id (here they are both the current 689 * context id) and the attach-only flag cannot exist during 690 * create. We also ensure specified peer is this context or 691 * an invalid one. 692 */ 693 if (queue_pair_entry->qp.handle.context != context_id || 694 (queue_pair_entry->qp.peer != VMCI_INVALID_ID && 695 queue_pair_entry->qp.peer != context_id)) { 696 result = VMCI_ERROR_NO_ACCESS; 697 goto error; 698 } 699 700 if (queue_pair_entry->qp.flags & VMCI_QPFLAG_ATTACH_ONLY) { 701 result = VMCI_ERROR_NOT_FOUND; 702 goto error; 703 } 704 } else { 705 result = vmci_queue_pair_alloc_hypercall(queue_pair_entry); 706 if (result < VMCI_SUCCESS) { 707 VMCI_LOG_WARNING( 708 LGPFX"vmci_queue_pair_alloc_hypercall result = " 709 "%d.\n", result); 710 goto error; 711 } 712 } 713 714 queue_pair_list_add_entry(&qp_guest_endpoints, &queue_pair_entry->qp); 715 716 out: 717 queue_pair_entry->qp.ref_count++; 718 *handle = queue_pair_entry->qp.handle; 719 *produce_q = (struct vmci_queue *)my_produce_q; 720 *consume_q = (struct vmci_queue *)my_consume_q; 721 722 /* 723 * We should initialize the queue pair header pages on a local queue 724 * pair create. For non-local queue pairs, the hypervisor initializes 725 * the header pages in the create step. 726 */ 727 if ((queue_pair_entry->qp.flags & VMCI_QPFLAG_LOCAL) && 728 queue_pair_entry->qp.ref_count == 1) { 729 vmci_queue_header_init((*produce_q)->q_header, *handle); 730 vmci_queue_header_init((*consume_q)->q_header, *handle); 731 } 732 733 vmci_mutex_release(&qp_guest_endpoints.mutex); 734 735 return (VMCI_SUCCESS); 736 737 error: 738 vmci_mutex_release(&qp_guest_endpoints.mutex); 739 if (queue_pair_entry) { 740 /* The queues will be freed inside the destroy routine. */ 741 qp_guest_endpoint_destroy(queue_pair_entry); 742 } else { 743 if (my_produce_q) 744 vmci_free_queue(my_produce_q, produce_size); 745 if (my_consume_q) 746 vmci_free_queue(my_consume_q, consume_size); 747 } 748 return (result); 749 750 error_keep_entry: 751 /* This path should only be used when an existing entry was found. */ 752 ASSERT(queue_pair_entry->qp.ref_count > 0); 753 vmci_mutex_release(&qp_guest_endpoints.mutex); 754 return (result); 755 } 756 757 /* 758 *------------------------------------------------------------------------------ 759 * 760 * vmci_queue_pair_detach_hypercall -- 761 * 762 * Helper to make a QueuePairDetach hypercall when the driver is supporting 763 * a guest device. 764 * 765 * Results: 766 * Result of the hypercall. 767 * 768 * Side effects: 769 * None. 770 * 771 *------------------------------------------------------------------------------ 772 */ 773 774 int 775 vmci_queue_pair_detach_hypercall(struct vmci_handle handle) 776 { 777 struct vmci_queue_pair_detach_msg detach_msg; 778 779 detach_msg.hdr.dst = VMCI_MAKE_HANDLE(VMCI_HYPERVISOR_CONTEXT_ID, 780 VMCI_QUEUEPAIR_DETACH); 781 detach_msg.hdr.src = VMCI_ANON_SRC_HANDLE; 782 detach_msg.hdr.payload_size = sizeof(handle); 783 detach_msg.handle = handle; 784 785 return (vmci_send_datagram((struct vmci_datagram *)&detach_msg)); 786 } 787 788 /* 789 *------------------------------------------------------------------------------ 790 * 791 * vmci_queue_pair_detach_guest_work -- 792 * 793 * Helper for VMCI QueuePair detach interface. Frees the physical pages for 794 * the queue pair. 795 * 796 * Results: 797 * Success or failure. 798 * 799 * Side effects: 800 * Memory may be freed. 801 * 802 *------------------------------------------------------------------------------ 803 */ 804 805 static int 806 vmci_queue_pair_detach_guest_work(struct vmci_handle handle) 807 { 808 struct qp_guest_endpoint *entry; 809 int result; 810 uint32_t ref_count; 811 812 ASSERT(!VMCI_HANDLE_INVALID(handle)); 813 814 vmci_mutex_acquire(&qp_guest_endpoints.mutex); 815 816 entry = (struct qp_guest_endpoint *)queue_pair_list_find_entry( 817 &qp_guest_endpoints, handle); 818 if (!entry) { 819 vmci_mutex_release(&qp_guest_endpoints.mutex); 820 return (VMCI_ERROR_NOT_FOUND); 821 } 822 823 ASSERT(entry->qp.ref_count >= 1); 824 825 if (entry->qp.flags & VMCI_QPFLAG_LOCAL) { 826 result = VMCI_SUCCESS; 827 828 if (entry->qp.ref_count > 1) { 829 result = queue_pair_notify_peer_local(false, handle); 830 831 /* 832 * We can fail to notify a local queuepair because we 833 * can't allocate. We still want to release the entry 834 * if that happens, so don't bail out yet. 835 */ 836 } 837 } else { 838 result = vmci_queue_pair_detach_hypercall(handle); 839 if (entry->hibernate_failure) { 840 if (result == VMCI_ERROR_NOT_FOUND) { 841 842 /* 843 * If a queue pair detach failed when entering 844 * hibernation, the guest driver and the device 845 * may disagree on its existence when coming 846 * out of hibernation. The guest driver will 847 * regard it as a non-local queue pair, but 848 * the device state is gone, since the device 849 * has been powered off. In this case, we 850 * treat the queue pair as a local queue pair 851 * with no peer. 852 */ 853 854 ASSERT(entry->qp.ref_count == 1); 855 result = VMCI_SUCCESS; 856 } 857 } 858 if (result < VMCI_SUCCESS) { 859 860 /* 861 * We failed to notify a non-local queuepair. That other 862 * queuepair might still be accessing the shared 863 * memory, so don't release the entry yet. It will get 864 * cleaned up by vmci_queue_pair_Exit() if necessary 865 * (assuming we are going away, otherwise why did this 866 * fail?). 867 */ 868 869 vmci_mutex_release(&qp_guest_endpoints.mutex); 870 return (result); 871 } 872 } 873 874 /* 875 * If we get here then we either failed to notify a local queuepair, or 876 * we succeeded in all cases. Release the entry if required. 877 */ 878 879 entry->qp.ref_count--; 880 if (entry->qp.ref_count == 0) 881 queue_pair_list_remove_entry(&qp_guest_endpoints, &entry->qp); 882 883 /* If we didn't remove the entry, this could change once we unlock. */ 884 ref_count = entry ? entry->qp.ref_count : 885 0xffffffff; /* 886 * Value does not matter, silence the 887 * compiler. 888 */ 889 890 vmci_mutex_release(&qp_guest_endpoints.mutex); 891 892 if (ref_count == 0) 893 qp_guest_endpoint_destroy(entry); 894 return (result); 895 } 896 897 /* 898 *------------------------------------------------------------------------------ 899 * 900 * queue_pair_notify_peer_local -- 901 * 902 * Dispatches a queue pair event message directly into the local event 903 * queue. 904 * 905 * Results: 906 * VMCI_SUCCESS on success, error code otherwise 907 * 908 * Side effects: 909 * None. 910 * 911 *------------------------------------------------------------------------------ 912 */ 913 914 static int 915 queue_pair_notify_peer_local(bool attach, struct vmci_handle handle) 916 { 917 struct vmci_event_msg *e_msg; 918 struct vmci_event_payload_qp *e_payload; 919 /* buf is only 48 bytes. */ 920 vmci_id context_id; 921 context_id = vmci_get_context_id(); 922 char buf[sizeof(*e_msg) + sizeof(*e_payload)]; 923 924 e_msg = (struct vmci_event_msg *)buf; 925 e_payload = vmci_event_msg_payload(e_msg); 926 927 e_msg->hdr.dst = VMCI_MAKE_HANDLE(context_id, VMCI_EVENT_HANDLER); 928 e_msg->hdr.src = VMCI_MAKE_HANDLE(VMCI_HYPERVISOR_CONTEXT_ID, 929 VMCI_CONTEXT_RESOURCE_ID); 930 e_msg->hdr.payload_size = sizeof(*e_msg) + sizeof(*e_payload) - 931 sizeof(e_msg->hdr); 932 e_msg->event_data.event = attach ? VMCI_EVENT_QP_PEER_ATTACH : 933 VMCI_EVENT_QP_PEER_DETACH; 934 e_payload->peer_id = context_id; 935 e_payload->handle = handle; 936 937 return (vmci_event_dispatch((struct vmci_datagram *)e_msg)); 938 } 939