aboutsummaryrefslogtreecommitdiffstats
path: root/drivers/virtio
diff options
context:
space:
mode:
Diffstat (limited to 'drivers/virtio')
-rw-r--r--drivers/virtio/virtio_pci.c125
1 files changed, 70 insertions, 55 deletions
diff --git a/drivers/virtio/virtio_pci.c b/drivers/virtio/virtio_pci.c
index 248e00ec4dc1..4a1f1ebff7bf 100644
--- a/drivers/virtio/virtio_pci.c
+++ b/drivers/virtio/virtio_pci.c
@@ -84,7 +84,7 @@ struct virtio_pci_vq_info
84 struct list_head node; 84 struct list_head node;
85 85
86 /* MSI-X vector (or none) */ 86 /* MSI-X vector (or none) */
87 unsigned vector; 87 unsigned msix_vector;
88}; 88};
89 89
90/* Qumranet donated their vendor ID for devices 0x1000 thru 0x10FF. */ 90/* Qumranet donated their vendor ID for devices 0x1000 thru 0x10FF. */
@@ -280,25 +280,14 @@ static void vp_free_vectors(struct virtio_device *vdev)
280 vp_dev->msix_entries = NULL; 280 vp_dev->msix_entries = NULL;
281} 281}
282 282
283static int vp_request_vectors(struct virtio_device *vdev, int nvectors, 283static int vp_request_msix_vectors(struct virtio_device *vdev, int nvectors,
284 bool per_vq_vectors) 284 bool per_vq_vectors)
285{ 285{
286 struct virtio_pci_device *vp_dev = to_vp_device(vdev); 286 struct virtio_pci_device *vp_dev = to_vp_device(vdev);
287 const char *name = dev_name(&vp_dev->vdev.dev); 287 const char *name = dev_name(&vp_dev->vdev.dev);
288 unsigned i, v; 288 unsigned i, v;
289 int err = -ENOMEM; 289 int err = -ENOMEM;
290 290
291 if (!nvectors) {
292 /* Can't allocate MSI-X vectors, use regular interrupt */
293 vp_dev->msix_vectors = 0;
294 err = request_irq(vp_dev->pci_dev->irq, vp_interrupt,
295 IRQF_SHARED, name, vp_dev);
296 if (err)
297 return err;
298 vp_dev->intx_enabled = 1;
299 return 0;
300 }
301
302 vp_dev->msix_entries = kmalloc(nvectors * sizeof *vp_dev->msix_entries, 291 vp_dev->msix_entries = kmalloc(nvectors * sizeof *vp_dev->msix_entries,
303 GFP_KERNEL); 292 GFP_KERNEL);
304 if (!vp_dev->msix_entries) 293 if (!vp_dev->msix_entries)
@@ -311,6 +300,7 @@ static int vp_request_vectors(struct virtio_device *vdev, int nvectors,
311 for (i = 0; i < nvectors; ++i) 300 for (i = 0; i < nvectors; ++i)
312 vp_dev->msix_entries[i].entry = i; 301 vp_dev->msix_entries[i].entry = i;
313 302
303 /* pci_enable_msix returns positive if we can't get this many. */
314 err = pci_enable_msix(vp_dev->pci_dev, vp_dev->msix_entries, nvectors); 304 err = pci_enable_msix(vp_dev->pci_dev, vp_dev->msix_entries, nvectors);
315 if (err > 0) 305 if (err > 0)
316 err = -ENOSPC; 306 err = -ENOSPC;
@@ -356,10 +346,22 @@ error:
356 return err; 346 return err;
357} 347}
358 348
359static struct virtqueue *vp_find_vq(struct virtio_device *vdev, unsigned index, 349static int vp_request_intx(struct virtio_device *vdev)
360 void (*callback)(struct virtqueue *vq), 350{
361 const char *name, 351 int err;
362 u16 vector) 352 struct virtio_pci_device *vp_dev = to_vp_device(vdev);
353
354 err = request_irq(vp_dev->pci_dev->irq, vp_interrupt,
355 IRQF_SHARED, dev_name(&vdev->dev), vp_dev);
356 if (!err)
357 vp_dev->intx_enabled = 1;
358 return err;
359}
360
361static struct virtqueue *setup_vq(struct virtio_device *vdev, unsigned index,
362 void (*callback)(struct virtqueue *vq),
363 const char *name,
364 u16 msix_vec)
363{ 365{
364 struct virtio_pci_device *vp_dev = to_vp_device(vdev); 366 struct virtio_pci_device *vp_dev = to_vp_device(vdev);
365 struct virtio_pci_vq_info *info; 367 struct virtio_pci_vq_info *info;
@@ -384,7 +386,7 @@ static struct virtqueue *vp_find_vq(struct virtio_device *vdev, unsigned index,
384 386
385 info->queue_index = index; 387 info->queue_index = index;
386 info->num = num; 388 info->num = num;
387 info->vector = vector; 389 info->msix_vector = msix_vec;
388 390
389 size = PAGE_ALIGN(vring_size(num, VIRTIO_PCI_VRING_ALIGN)); 391 size = PAGE_ALIGN(vring_size(num, VIRTIO_PCI_VRING_ALIGN));
390 info->queue = alloc_pages_exact(size, GFP_KERNEL|__GFP_ZERO); 392 info->queue = alloc_pages_exact(size, GFP_KERNEL|__GFP_ZERO);
@@ -408,10 +410,10 @@ static struct virtqueue *vp_find_vq(struct virtio_device *vdev, unsigned index,
408 vq->priv = info; 410 vq->priv = info;
409 info->vq = vq; 411 info->vq = vq;
410 412
411 if (vector != VIRTIO_MSI_NO_VECTOR) { 413 if (msix_vec != VIRTIO_MSI_NO_VECTOR) {
412 iowrite16(vector, vp_dev->ioaddr + VIRTIO_MSI_QUEUE_VECTOR); 414 iowrite16(msix_vec, vp_dev->ioaddr + VIRTIO_MSI_QUEUE_VECTOR);
413 vector = ioread16(vp_dev->ioaddr + VIRTIO_MSI_QUEUE_VECTOR); 415 msix_vec = ioread16(vp_dev->ioaddr + VIRTIO_MSI_QUEUE_VECTOR);
414 if (vector == VIRTIO_MSI_NO_VECTOR) { 416 if (msix_vec == VIRTIO_MSI_NO_VECTOR) {
415 err = -EBUSY; 417 err = -EBUSY;
416 goto out_assign; 418 goto out_assign;
417 } 419 }
@@ -472,7 +474,8 @@ static void vp_del_vqs(struct virtio_device *vdev)
472 list_for_each_entry_safe(vq, n, &vdev->vqs, list) { 474 list_for_each_entry_safe(vq, n, &vdev->vqs, list) {
473 info = vq->priv; 475 info = vq->priv;
474 if (vp_dev->per_vq_vectors) 476 if (vp_dev->per_vq_vectors)
475 free_irq(vp_dev->msix_entries[info->vector].vector, vq); 477 free_irq(vp_dev->msix_entries[info->msix_vector].vector,
478 vq);
476 vp_del_vq(vq); 479 vp_del_vq(vq);
477 } 480 }
478 vp_dev->per_vq_vectors = false; 481 vp_dev->per_vq_vectors = false;
@@ -484,38 +487,58 @@ static int vp_try_to_find_vqs(struct virtio_device *vdev, unsigned nvqs,
484 struct virtqueue *vqs[], 487 struct virtqueue *vqs[],
485 vq_callback_t *callbacks[], 488 vq_callback_t *callbacks[],
486 const char *names[], 489 const char *names[],
487 int nvectors, 490 bool use_msix,
488 bool per_vq_vectors) 491 bool per_vq_vectors)
489{ 492{
490 struct virtio_pci_device *vp_dev = to_vp_device(vdev); 493 struct virtio_pci_device *vp_dev = to_vp_device(vdev);
491 u16 vector; 494 u16 msix_vec;
492 int i, err, allocated_vectors; 495 int i, err, nvectors, allocated_vectors;
493 496
494 err = vp_request_vectors(vdev, nvectors, per_vq_vectors); 497 if (!use_msix) {
495 if (err) 498 /* Old style: one normal interrupt for change and all vqs. */
496 goto error_request; 499 err = vp_request_intx(vdev);
500 if (err)
501 goto error_request;
502 } else {
503 if (per_vq_vectors) {
504 /* Best option: one for change interrupt, one per vq. */
505 nvectors = 1;
506 for (i = 0; i < nvqs; ++i)
507 if (callbacks[i])
508 ++nvectors;
509 } else {
510 /* Second best: one for change, shared for all vqs. */
511 nvectors = 2;
512 }
513
514 err = vp_request_msix_vectors(vdev, nvectors, per_vq_vectors);
515 if (err)
516 goto error_request;
517 }
497 518
498 vp_dev->per_vq_vectors = per_vq_vectors; 519 vp_dev->per_vq_vectors = per_vq_vectors;
499 allocated_vectors = vp_dev->msix_used_vectors; 520 allocated_vectors = vp_dev->msix_used_vectors;
500 for (i = 0; i < nvqs; ++i) { 521 for (i = 0; i < nvqs; ++i) {
501 if (!callbacks[i] || !vp_dev->msix_enabled) 522 if (!callbacks[i] || !vp_dev->msix_enabled)
502 vector = VIRTIO_MSI_NO_VECTOR; 523 msix_vec = VIRTIO_MSI_NO_VECTOR;
503 else if (vp_dev->per_vq_vectors) 524 else if (vp_dev->per_vq_vectors)
504 vector = allocated_vectors++; 525 msix_vec = allocated_vectors++;
505 else 526 else
506 vector = VP_MSIX_VQ_VECTOR; 527 msix_vec = VP_MSIX_VQ_VECTOR;
507 vqs[i] = vp_find_vq(vdev, i, callbacks[i], names[i], vector); 528 vqs[i] = setup_vq(vdev, i, callbacks[i], names[i], msix_vec);
508 if (IS_ERR(vqs[i])) { 529 if (IS_ERR(vqs[i])) {
509 err = PTR_ERR(vqs[i]); 530 err = PTR_ERR(vqs[i]);
510 goto error_find; 531 goto error_find;
511 } 532 }
512 /* allocate per-vq irq if available and necessary */ 533 /* allocate per-vq irq if available and necessary */
513 if (vp_dev->per_vq_vectors && vector != VIRTIO_MSI_NO_VECTOR) { 534 if (vp_dev->per_vq_vectors) {
514 snprintf(vp_dev->msix_names[vector], sizeof *vp_dev->msix_names, 535 snprintf(vp_dev->msix_names[msix_vec],
515 "%s-%s", dev_name(&vp_dev->vdev.dev), names[i]); 536 sizeof *vp_dev->msix_names,
516 err = request_irq(vp_dev->msix_entries[vector].vector, 537 "%s-%s",
517 vring_interrupt, 0, 538 dev_name(&vp_dev->vdev.dev), names[i]);
518 vp_dev->msix_names[vector], vqs[i]); 539 err = request_irq(msix_vec, vring_interrupt, 0,
540 vp_dev->msix_names[msix_vec],
541 vqs[i]);
519 if (err) { 542 if (err) {
520 vp_del_vq(vqs[i]); 543 vp_del_vq(vqs[i]);
521 goto error_find; 544 goto error_find;
@@ -537,28 +560,20 @@ static int vp_find_vqs(struct virtio_device *vdev, unsigned nvqs,
537 vq_callback_t *callbacks[], 560 vq_callback_t *callbacks[],
538 const char *names[]) 561 const char *names[])
539{ 562{
540 int vectors = 0; 563 int err;
541 int i, uninitialized_var(err);
542
543 /* How many vectors would we like? */
544 for (i = 0; i < nvqs; ++i)
545 if (callbacks[i])
546 ++vectors;
547 564
548 /* We want at most one vector per queue and one for config changes. */ 565 /* Try MSI-X with one vector per queue. */
549 err = vp_try_to_find_vqs(vdev, nvqs, vqs, callbacks, names, 566 err = vp_try_to_find_vqs(vdev, nvqs, vqs, callbacks, names, true, true);
550 vectors + 1, true);
551 if (!err) 567 if (!err)
552 return 0; 568 return 0;
553 /* Fallback to separate vectors for config and a shared for queues. */ 569 /* Fallback: MSI-X with one vector for config, one shared for queues. */
554 err = vp_try_to_find_vqs(vdev, nvqs, vqs, callbacks, names, 570 err = vp_try_to_find_vqs(vdev, nvqs, vqs, callbacks, names,
555 2, false); 571 true, false);
556 if (!err) 572 if (!err)
557 return 0; 573 return 0;
558 /* Finally fall back to regular interrupts. */ 574 /* Finally fall back to regular interrupts. */
559 err = vp_try_to_find_vqs(vdev, nvqs, vqs, callbacks, names, 575 return vp_try_to_find_vqs(vdev, nvqs, vqs, callbacks, names,
560 0, false); 576 false, false);
561 return err;
562} 577}
563 578
564static struct virtio_config_ops virtio_pci_config_ops = { 579static struct virtio_config_ops virtio_pci_config_ops = {