VirtualBox

source: vbox/trunk/src/VBox/Additions/solaris/Virtio/VirtioPci-solaris.c@ 57260

Last change on this file since 57260 was 55980, checked in by vboxsync, 10 years ago

iprt/log.h,++: Added extended logger instance getters that also checks whether the given logger and group-flags are enabled, making the LogRel* checks more efficient in avoid uncessary RTLogLoggerEx parameter building and calls. Ditto for debug logging. The LOG_INSTANCE and LOG_REL_INSTANCE tricks are gone for now.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 23.8 KB
Line 
1/* $Id: VirtioPci-solaris.c 55980 2015-05-20 17:35:22Z vboxsync $ */
2/** @file
3 * VirtualBox Guest Additions - Virtio Driver for Solaris, PCI Hypervisor Interface.
4 */
5
6/*
7 * Copyright (C) 2010-2011 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 *
17 * The contents of this file may alternatively be used under the terms
18 * of the Common Development and Distribution License Version 1.0
19 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the
20 * VirtualBox OSE distribution, in which case the provisions of the
21 * CDDL are applicable instead of those of the GPL.
22 *
23 * You may elect to license modified versions of this file under the
24 * terms and conditions of either the GPL or the CDDL or both.
25 */
26
27/*******************************************************************************
28* Header Files *
29*******************************************************************************/
30#include "VirtioPci-solaris.h"
31
32#include <iprt/asm.h>
33#include <iprt/assert.h>
34#include <iprt/mem.h>
35#include <iprt/param.h>
36#include <VBox/log.h>
37
38#include <sys/pci.h>
39#include <sys/param.h>
40
41/*******************************************************************************
42* Defined Constants And Macros *
43*******************************************************************************/
44/*
45 * Pci Register offsets.
46 */
47#define VIRTIO_PCI_HOST_FEATURES 0x00
48#define VIRTIO_PCI_GUEST_FEATURES 0x04
49#define VIRTIO_PCI_QUEUE_PFN 0x08
50#define VIRTIO_PCI_QUEUE_NUM 0x0C
51#define VIRTIO_PCI_QUEUE_SEL 0x0E
52#define VIRTIO_PCI_QUEUE_NOTIFY 0x10
53#define VIRTIO_PCI_STATUS 0x12
54#define VIRTIO_PCI_ISR 0x13
55#define VIRTIO_PCI_CONFIG 0x14
56
57#define VIRTIO_PCI_RING_ALIGN PAGE_SIZE
58#define VIRTIO_PCI_QUEUE_ADDR_SHIFT PAGE_SHIFT
59
60/**
61 * virtio_pci_t: Private data per device instance.
62 */
63typedef struct virtio_pci_t
64{
65 ddi_acc_handle_t hIO; /* IO handle */
66 caddr_t addrIOBase; /* IO base address */
67} virtio_pci_t;
68
69/**
70 * virtio_pci_queue_t: Private data per queue instance.
71 */
72typedef struct virtio_pci_queue_t
73{
74 ddi_dma_handle_t hDMA; /* DMA handle. */
75 ddi_acc_handle_t hIO; /* IO handle. */
76 size_t cbBuf; /* Physical address of buffer. */
77 paddr_t physBuf; /* Size of buffer. */
78 pfn_t pageBuf; /* Page frame number of buffer. */
79} virtio_pci_queue_t;
80
81static ddi_device_acc_attr_t g_VirtioPciAccAttrRegs =
82{
83 DDI_DEVICE_ATTR_V0, /* Version */
84 DDI_STRUCTURE_LE_ACC, /* Structural data access in little endian. */
85 DDI_STRICTORDER_ACC, /* Strict ordering. */
86 DDI_DEFAULT_ACC /* Default access, possible panic on errors*/
87};
88
89static ddi_device_acc_attr_t g_VirtioPciAccAttrRing =
90{
91 DDI_DEVICE_ATTR_V0, /* Version. */
92 DDI_NEVERSWAP_ACC, /* Data access with no byte swapping*/
93 DDI_STRICTORDER_ACC, /* Strict ordering. */
94 DDI_DEFAULT_ACC /* Default access, possible panic on errors*/
95};
96
97static ddi_dma_attr_t g_VirtioPciDmaAttrRing =
98{
99 DMA_ATTR_V0, /* Version. */
100 0, /* Lowest usable address. */
101 0xffffffffffffffffULL, /* Highest usable address. */
102 0x7fffffff, /* Maximum DMAable byte count. */
103 VIRTIO_PCI_RING_ALIGN, /* Alignment in bytes. */
104 0x7ff, /* Bitmap of burst sizes */
105 1, /* Minimum transfer. */
106 0xffffffffU, /* Maximum transfer. */
107 0xffffffffffffffffULL, /* Maximum segment length. */
108 1, /* Maximum number of segments. */
109 1, /* Granularity. */
110 0 /* Flags (reserved). */
111};
112
113/** Pointer to the interrupt handle vector */
114static ddi_intr_handle_t *g_pIntr;
115/** Number of actually allocated interrupt handles */
116static size_t g_cIntrAllocated;
117/** The IRQ Mutex */
118static kmutex_t g_IrqMtx;
119
120
121/*******************************************************************************
122* Internal Functions *
123*******************************************************************************/
124static void *VirtioPciAlloc(PVIRTIODEVICE pDevice);
125static void VirtioPciFree(PVIRTIODEVICE pDevice);
126static int VirtioPciAttach(PVIRTIODEVICE pDevice);
127static int VirtioPciDetach(PVIRTIODEVICE pDevice);
128static uint32_t VirtioPciGetFeatures(PVIRTIODEVICE pDevice);
129static void VirtioPciSetFeatures(PVIRTIODEVICE pDevice, uint32_t fFeatures);
130static int VirtioPciNotifyQueue(PVIRTIODEVICE pDevice, PVIRTIOQUEUE pQueue);
131static void VirtioPciGet(PVIRTIODEVICE pDevice, off_t off, void *pv, size_t cb);
132static void VirtioPciSet(PVIRTIODEVICE pDevice, off_t off, void *pv, size_t cb);
133static void *VirtioPciGetQueue(PVIRTIODEVICE pDevice, PVIRTIOQUEUE pQueue);
134static void VirtioPciPutQueue(PVIRTIODEVICE pDevice, PVIRTIOQUEUE pQueue);
135static void VirtioPciSetStatus(PVIRTIODEVICE pDevice, uint8_t Status);
136
137static uint_t VirtioPciISR(caddr_t Arg);
138static int VirtioPciSetupIRQ(dev_info_t *pDip);
139static void VirtioPciRemoveIRQ(dev_info_t *pDip);
140
141/**
142 * Hypervisor operations for Virtio Pci.
143 */
144VIRTIOHYPEROPS g_VirtioHyperOpsPci =
145{
146 VirtioPciAlloc,
147 VirtioPciFree,
148 VirtioPciAttach,
149 VirtioPciDetach,
150 VirtioPciGetFeatures,
151 VirtioPciSetFeatures,
152 VirtioPciNotifyQueue,
153 VirtioPciGet,
154 VirtioPciSet,
155 VirtioPciGetQueue,
156 VirtioPciPutQueue,
157 VirtioPciSetStatus
158};
159
160
161/**
162 * Virtio Pci private data allocation routine.
163 *
164 * @param pDevice Pointer to the Virtio device instance.
165 * @return Allocated private data structure which must only be freed by calling
166 * VirtioPciFree().
167 */
168static void *VirtioPciAlloc(PVIRTIODEVICE pDevice)
169{
170 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciAlloc pDevice=%p\n", pDevice));
171 virtio_pci_t *pPciData = RTMemAllocZ(sizeof(virtio_pci_t));
172 return pPciData;
173}
174
175
176/**
177 * Virtio Pci private data deallocation routine.
178 *
179 * @param pDevice Pointer to the Virtio device instance.
180 */
181static void VirtioPciFree(PVIRTIODEVICE pDevice)
182{
183 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciFree pDevice=%p\n", pDevice));
184 virtio_pci_t *pPciData = pDevice->pvHyper;
185 if (pPciData)
186 {
187 RTMemFree(pDevice->pvHyper);
188 pDevice->pvHyper = NULL;
189 }
190}
191
192
193/**
194 * Virtio Pci attach routine, called from driver attach.
195 *
196 * @param pDevice Pointer to the Virtio device instance.
197 *
198 * @return Solaris DDI error code. DDI_SUCCESS or DDI_FAILURE.
199 */
200static int VirtioPciAttach(PVIRTIODEVICE pDevice)
201{
202 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciAttach pDevice=%p\n", pDevice));
203 virtio_pci_t *pPciData = pDevice->pvHyper;
204 AssertReturn(pPciData, DDI_FAILURE);
205
206 int rc = ddi_regs_map_setup(pDevice->pDip,
207 1, /* reg. num */
208 &pPciData->addrIOBase,
209 0, /* offset */
210 0, /* length */
211 &g_VirtioPciAccAttrRegs,
212 &pPciData->hIO);
213 if (rc == DDI_SUCCESS)
214 {
215 /*
216 * Reset the device.
217 */
218 VirtioPciSetStatus(pDevice, 0);
219
220 /*
221 * Add interrupt handler.
222 */
223 VirtioPciSetupIRQ(pDevice->pDip);
224
225 LogFlow((VIRTIOLOGNAME ":VirtioPciAttach: successfully mapped registers.\n"));
226 return DDI_SUCCESS;
227 }
228 else
229 LogRel((VIRTIOLOGNAME ":VirtioPciAttach: ddi_regs_map_setup failed. rc=%d\n", rc));
230 return DDI_FAILURE;
231}
232
233
234/**
235 * Virtio Pci detach routine, called from driver detach.
236 *
237 * @param pDevice Pointer to the Virtio device instance.
238 *
239 * @return Solaris DDI error code. DDI_SUCCESS or DDI_FAILURE.
240 */
241static int VirtioPciDetach(PVIRTIODEVICE pDevice)
242{
243 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciDetach pDevice=%p\n", pDevice));
244 virtio_pci_t *pPciData = pDevice->pvHyper;
245 AssertReturn(pPciData, DDI_FAILURE);
246
247 VirtioPciRemoveIRQ(pDevice->pDip);
248 ddi_regs_map_free(&pPciData->hIO);
249 return DDI_SUCCESS;
250}
251
252
253/**
254 * Get host supported features.
255 *
256 * @param pDevice Pointer to the Virtio device instance.
257 *
258 * @return Mask of host features.
259 */
260static uint32_t VirtioPciGetFeatures(PVIRTIODEVICE pDevice)
261{
262 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciGetFeatures pDevice=%p\n", pDevice));
263 virtio_pci_t *pPciData = pDevice->pvHyper;
264 AssertReturn(pPciData, 0);
265
266 return ddi_get32(pPciData->hIO, (uint32_t *)(pPciData->addrIOBase + VIRTIO_PCI_HOST_FEATURES));
267}
268
269
270/**
271 * Set guest supported features.
272 *
273 * @param pDevice Pointer to the Virtio device instance.
274 * @param u32Features Mask of guest supported features.
275 */
276static void VirtioPciSetFeatures(PVIRTIODEVICE pDevice, uint32_t u32Features)
277{
278 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciSetFeatures pDevice=%p\n", pDevice));
279 virtio_pci_t *pPciData = pDevice->pvHyper;
280 AssertReturnVoid(pPciData);
281
282 ddi_put32(pPciData->hIO, (uint32_t *)(pPciData->addrIOBase + VIRTIO_PCI_GUEST_FEATURES), u32Features);
283}
284
285
286/**
287 * Update the queue, notify the host.
288 *
289 * @param pDevice Pointer to the Virtio device instance.
290 * @param pQueue Pointer to the Queue that is doing the notification.
291 *
292 * @return Solaris DDI error code. DDI_SUCCESS or DDI_FAILURE.
293 */
294static int VirtioPciNotifyQueue(PVIRTIODEVICE pDevice, PVIRTIOQUEUE pQueue)
295{
296 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciNotifyQueue pDevice=%p pQueue=%p\n", pDevice, pQueue));
297 virtio_pci_t *pPciData = pDevice->pvHyper;
298 AssertReturn(pPciData, DDI_FAILURE);
299
300 pQueue->Ring.pRingAvail->Index += pQueue->cBufs;
301 pQueue->cBufs = 0;
302
303 ASMCompilerBarrier();
304
305 ddi_put16(pPciData->hIO, (uint16_t *)(pPciData->addrIOBase + VIRTIO_PCI_QUEUE_NOTIFY), pQueue->QueueIndex);
306 cmn_err(CE_NOTE, "VirtioPciNotifyQueue\n");
307}
308
309
310
311/**
312 * Virtio Pci set (write) routine.
313 *
314 * @param pDevice Pointer to the Virtio device instance.
315 * @param off Offset into the PCI config space.
316 * @param pv Pointer to the buffer to write from.
317 * @param cb Size of the buffer in bytes.
318 */
319static void VirtioPciSet(PVIRTIODEVICE pDevice, off_t off, void *pv, size_t cb)
320{
321 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciSet pDevice=%p\n", pDevice));
322 virtio_pci_t *pPciData = pDevice->pvHyper;
323 AssertReturnVoid(pPciData);
324
325 uint8_t *pb = pv;
326 for (size_t i = 0; i < cb; i++, pb++)
327 ddi_put8(pPciData->hIO, (uint8_t *)(pPciData->addrIOBase + VIRTIO_PCI_CONFIG + off + i), *pb);
328}
329
330
331/**
332 * Virtio Pci get (read) routine.
333 *
334 * @param pDevice Pointer to the Virtio device instance.
335 * @param off Offset into the PCI config space.
336 * @param pv Where to store the read data.
337 * @param cb Size of the buffer in bytes.
338 */
339static void VirtioPciGet(PVIRTIODEVICE pDevice, off_t off, void *pv, size_t cb)
340{
341 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciGet pDevice=%p off=%u pv=%p cb=%u\n", pDevice, off, pv, cb));
342 virtio_pci_t *pPciData = pDevice->pvHyper;
343 AssertReturnVoid(pPciData);
344
345 uint8_t *pb = pv;
346 for (size_t i = 0; i < cb; i++, pb++)
347 *pb = ddi_get8(pPciData->hIO, (uint8_t *)(pPciData->addrIOBase + VIRTIO_PCI_CONFIG + off + i));
348}
349
350
351/**
352 * Virtio Pci put queue routine. Places the queue and frees associated queue.
353 *
354 * @param pDevice Pointer to the Virtio device instance.
355 * @param pQueue Pointer to the queue.
356 */
357static void VirtioPciPutQueue(PVIRTIODEVICE pDevice, PVIRTIOQUEUE pQueue)
358{
359 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciPutQueue pDevice=%p pQueue=%p\n", pDevice, pQueue));
360 AssertReturnVoid(pDevice);
361 AssertReturnVoid(pQueue);
362
363 virtio_pci_t *pPci = pDevice->pvHyper;
364 AssertReturnVoid(pPci);
365 virtio_pci_queue_t *pPciQueue = pQueue->pvData;
366 if (RT_UNLIKELY(!pPciQueue))
367 {
368 LogRel((VIRTIOLOGNAME ":VirtioPciPutQueue missing Pci queue.\n"));
369 return;
370 }
371
372 ddi_put16(pPci->hIO, (uint16_t *)(pPci->addrIOBase + VIRTIO_PCI_QUEUE_SEL), pQueue->QueueIndex);
373 ddi_put32(pPci->hIO, (uint32_t *)(pPci->addrIOBase + VIRTIO_PCI_QUEUE_PFN), 0);
374
375 ddi_dma_unbind_handle(pPciQueue->hDMA);
376 ddi_dma_mem_free(&pPciQueue->hIO);
377 ddi_dma_free_handle(&pPciQueue->hDMA);
378 RTMemFree(pPciQueue);
379}
380
381
382/**
383 * Virtio Pci get queue routine. Allocates a PCI queue and DMA resources.
384 *
385 * @param pDevice Pointer to the Virtio device instance.
386 * @param pQueue Where to store the queue.
387 *
388 * @return An allocated Virtio Pci queue, or NULL in case of errors.
389 */
390static void *VirtioPciGetQueue(PVIRTIODEVICE pDevice, PVIRTIOQUEUE pQueue)
391{
392 LogFlowFunc((VIRTIOLOGNAME ":VirtioPciGetQueue pDevice=%p pQueue=%p\n", pDevice, pQueue));
393 AssertReturn(pDevice, NULL);
394
395 virtio_pci_t *pPci = pDevice->pvHyper;
396 AssertReturn(pPci, NULL);
397
398 /*
399 * Select a Queue.
400 */
401 ddi_put16(pPci->hIO, (uint16_t *)(pPci->addrIOBase + VIRTIO_PCI_QUEUE_SEL), pQueue->QueueIndex);
402
403 /*
404 * Get the currently selected Queue's size.
405 */
406 pQueue->Ring.cDesc = ddi_get16(pPci->hIO, (uint16_t *)(pPci->addrIOBase + VIRTIO_PCI_QUEUE_NUM));
407 if (RT_UNLIKELY(!pQueue->Ring.cDesc))
408 {
409 LogRel((VIRTIOLOGNAME ": VirtioPciGetQueue: Queue[%d] has no descriptors.\n", pQueue->QueueIndex));
410 return NULL;
411 }
412
413 /*
414 * Check if it's already active.
415 */
416 uint32_t QueuePFN = ddi_get32(pPci->hIO, (uint32_t *)(pPci->addrIOBase + VIRTIO_PCI_QUEUE_PFN));
417 if (QueuePFN != 0)
418 {
419 LogRel((VIRTIOLOGNAME ":VirtioPciGetQueue: Queue[%d] is already used.\n", pQueue->QueueIndex));
420 return NULL;
421 }
422
423 LogFlow(("Queue[%d] has %d slots.\n", pQueue->QueueIndex, pQueue->Ring.cDesc));
424
425 /*
426 * Allocate and initialize Pci queue data.
427 */
428 virtio_pci_queue_t *pPciQueue = RTMemAllocZ(sizeof(virtio_pci_queue_t));
429 if (pPciQueue)
430 {
431 /*
432 * Setup DMA.
433 */
434 size_t cbQueue = VirtioRingSize(pQueue->Ring.cDesc, VIRTIO_PCI_RING_ALIGN);
435 int rc = ddi_dma_alloc_handle(pDevice->pDip, &g_VirtioPciDmaAttrRing, DDI_DMA_SLEEP, 0 /* addr */, &pPciQueue->hDMA);
436 if (rc == DDI_SUCCESS)
437 {
438 rc = ddi_dma_mem_alloc(pPciQueue->hDMA, cbQueue, &g_VirtioPciAccAttrRing, DDI_DMA_CONSISTENT,
439 DDI_DMA_SLEEP, 0 /* addr */, &pQueue->pQueue, &pPciQueue->cbBuf,
440 &pPciQueue->hIO);
441 if (rc == DDI_SUCCESS)
442 {
443 AssertRelease(pPciQueue->cbBuf >= cbQueue);
444 ddi_dma_cookie_t DmaCookie;
445 uint_t cCookies;
446 rc = ddi_dma_addr_bind_handle(pPciQueue->hDMA, NULL /* addrspace */, pQueue->pQueue, pPciQueue->cbBuf,
447 DDI_DMA_RDWR | DDI_DMA_CONSISTENT, DDI_DMA_SLEEP,
448 0 /* addr */, &DmaCookie, &cCookies);
449 if (rc == DDI_SUCCESS)
450 {
451 pPciQueue->physBuf = DmaCookie.dmac_laddress;
452 pPciQueue->pageBuf = pPciQueue->physBuf >> VIRTIO_PCI_QUEUE_ADDR_SHIFT;
453
454 LogFlow((VIRTIOLOGNAME ":VirtioPciGetQueue: Queue[%d]%p physBuf=%x pfn of Buf %#x\n", pQueue->QueueIndex,
455 pQueue->pQueue, pPciQueue->physBuf, pPciQueue->pageBuf));
456 cmn_err(CE_NOTE, ":VirtioPciGetQueue: Queue[%d]%p physBuf=%x pfn of Buf %x\n", pQueue->QueueIndex,
457 pQueue->pQueue, pPciQueue->physBuf, pPciQueue->pageBuf);
458
459 /*
460 * Activate the queue and initialize a ring for the queue.
461 */
462 memset(pQueue->pQueue, 0, pPciQueue->cbBuf);
463 ddi_put32(pPci->hIO, (uint32_t *)(pPci->addrIOBase + VIRTIO_PCI_QUEUE_PFN), pPciQueue->pageBuf);
464 VirtioRingInit(pQueue, pQueue->Ring.cDesc, pQueue->pQueue, VIRTIO_PCI_RING_ALIGN);
465 return pPciQueue;
466 }
467 else
468 LogRel((VIRTIOLOGNAME ":VirtioPciGetQueue: ddi_dma_addr_bind_handle failed. rc=%d\n", rc));
469
470 ddi_dma_mem_free(&pPciQueue->hIO);
471 }
472 else
473 LogRel((VIRTIOLOGNAME ":VirtioPciGetQueue: ddi_dma_mem_alloc failed for %u bytes rc=%d\n", cbQueue, rc));
474
475 ddi_dma_free_handle(&pPciQueue->hDMA);
476 }
477 else
478 LogRel((VIRTIOLOGNAME ":VirtioPciGetQueue: ddi_dma_alloc_handle failed. rc=%d\n", rc));
479
480 RTMemFree(pPciQueue);
481 }
482 else
483 LogRel((VIRTIOLOGNAME ":VirtioPciGetQueue: failed to alloc %u bytes for Pci Queue data.\n", sizeof(virtio_pci_queue_t)));
484
485 return NULL;
486}
487
488
489/**
490 * Set the Virtio PCI status bit.
491 *
492 * @param pDevice Pointer to the Virtio device instance.
493 * @param Status The status to set.
494 */
495static void VirtioPciSetStatus(PVIRTIODEVICE pDevice, uint8_t Status)
496{
497 virtio_pci_t *pPciData = pDevice->pvHyper;
498 ddi_put8(pPciData->hIO, (uint8_t *)(pPciData->addrIOBase + VIRTIO_PCI_STATUS), Status);
499}
500
501
502/**
503 * Sets up IRQ for Virtio PCI.
504 *
505 * @param pDip Pointer to the device info structure.
506 *
507 * @return Solaris error code.
508 */
509static int VirtioPciSetupIRQ(dev_info_t *pDip)
510{
511 LogFlow((VIRTIOLOGNAME ":VirtioPciSetupIRQ: pDip=%p\n", pDip));
512 cmn_err(CE_NOTE, "VirtioPciSetupIRQ\n");
513
514 int IntrType = 0;
515 int rc = ddi_intr_get_supported_types(pDip, &IntrType);
516 if (rc == DDI_SUCCESS)
517 {
518 /* We won't need to bother about MSIs. */
519 if (IntrType & DDI_INTR_TYPE_FIXED)
520 {
521 int IntrCount = 0;
522 rc = ddi_intr_get_nintrs(pDip, IntrType, &IntrCount);
523 if ( rc == DDI_SUCCESS
524 && IntrCount > 0)
525 {
526 int IntrAvail = 0;
527 rc = ddi_intr_get_navail(pDip, IntrType, &IntrAvail);
528 if ( rc == DDI_SUCCESS
529 && IntrAvail > 0)
530 {
531 /* Allocated kernel memory for the interrupt handles. The allocation size is stored internally. */
532 g_pIntr = RTMemAllocZ(IntrCount * sizeof(ddi_intr_handle_t));
533 if (g_pIntr)
534 {
535 int IntrAllocated;
536 rc = ddi_intr_alloc(pDip, g_pIntr, IntrType, 0, IntrCount, &IntrAllocated, DDI_INTR_ALLOC_NORMAL);
537 if ( rc == DDI_SUCCESS
538 && IntrAllocated > 0)
539 {
540 g_cIntrAllocated = IntrAllocated;
541 uint_t uIntrPriority;
542 rc = ddi_intr_get_pri(g_pIntr[0], &uIntrPriority);
543 if (rc == DDI_SUCCESS)
544 {
545 /* Initialize the mutex. */
546 mutex_init(&g_IrqMtx, NULL, MUTEX_DRIVER, DDI_INTR_PRI(uIntrPriority));
547
548 /* Assign interrupt handler functions and enable interrupts. */
549 for (int i = 0; i < IntrAllocated; i++)
550 {
551 rc = ddi_intr_add_handler(g_pIntr[i], (ddi_intr_handler_t *)VirtioPciISR,
552 NULL /* No Private Data */, NULL);
553 if (rc == DDI_SUCCESS)
554 rc = ddi_intr_enable(g_pIntr[i]);
555 if (rc != DDI_SUCCESS)
556 {
557 /* Changing local IntrAllocated to hold so-far allocated handles for freeing. */
558 IntrAllocated = i;
559 break;
560 }
561 }
562 if (rc == DDI_SUCCESS)
563 {
564 cmn_err(CE_NOTE, "VirtioPciSetupIRQ success\n");
565 return rc;
566 }
567
568 /* Remove any assigned handlers */
569 LogRel((VIRTIOLOGNAME ":VirtioPciSetupIRQ failed to assign IRQs allocated=%d\n", IntrAllocated));
570 for (int x = 0; x < IntrAllocated; x++)
571 ddi_intr_remove_handler(g_pIntr[x]);
572 }
573 else
574 LogRel((VIRTIOLOGNAME ":VirtioPciSetupIRQ failed to get priority of interrupt. rc=%d\n", rc));
575
576 /* Remove allocated IRQs, too bad we can free only one handle at a time. */
577 for (int k = 0; k < g_cIntrAllocated; k++)
578 ddi_intr_free(g_pIntr[k]);
579 }
580 else
581 LogRel((VIRTIOLOGNAME ":VirtioPciSetupIRQ: failed to allocated IRQs. count=%d\n", IntrCount));
582 RTMemFree(g_pIntr);
583 }
584 else
585 LogRel((VIRTIOLOGNAME ":VirtioPciSetupIRQ: failed to allocated IRQs. count=%d\n", IntrCount));
586 }
587 else
588 {
589 LogRel((VIRTIOLOGNAME ":VirtioPciSetupIRQ: failed to get or insufficient available IRQs. rc=%d IntrAvail=%d\n",
590 rc, IntrAvail));
591 }
592 }
593 else
594 {
595 LogRel((VIRTIOLOGNAME ":VirtioPciSetupIRQ: failed to get or insufficient number of IRQs. rc=%d IntrCount=%d\n", rc,
596 IntrCount));
597 }
598 }
599 else
600 LogRel((VIRTIOLOGNAME ":VirtioPciSetupIRQ: invalid irq type. IntrType=%#x\n", IntrType));
601 }
602 else
603 LogRel((VIRTIOLOGNAME ":VirtioPciSetupIRQ: failed to get supported interrupt types\n"));
604 return rc;
605}
606
607
608/**
609 * Removes IRQ for Virtio PCI device.
610 *
611 * @param pDip Pointer to the device info structure.
612 */
613static void VirtioPciRemoveIRQ(dev_info_t *pDip)
614{
615 LogFlow((VIRTIOLOGNAME ":VirtioPciRemoveIRQ pDip=%p:\n", pDip));
616
617 for (int i = 0; i < g_cIntrAllocated; i++)
618 {
619 int rc = ddi_intr_disable(g_pIntr[i]);
620 if (rc == DDI_SUCCESS)
621 {
622 rc = ddi_intr_remove_handler(g_pIntr[i]);
623 if (rc == DDI_SUCCESS)
624 ddi_intr_free(g_pIntr[i]);
625 }
626 }
627 RTMemFree(g_pIntr);
628 mutex_destroy(&g_IrqMtx);
629}
630
631
632/**
633 * Interrupt Service Routine for Virtio PCI device.
634 *
635 * @param Arg Private data (unused, will be NULL).
636 * @returns DDI_INTR_CLAIMED if it's our interrupt, DDI_INTR_UNCLAIMED if it isn't.
637 */
638static uint_t VirtioPciISR(caddr_t Arg)
639{
640 LogFlow((VIRTIOLOGNAME ":VBoxGuestSolarisISR\n"));
641 cmn_err(CE_NOTE, "VBoxGuestSolarisISRd Arg=%p\n", Arg);
642
643 mutex_enter(&g_IrqMtx);
644 bool fOurIRQ = false;
645 /*
646 * Call the DeviceOps ISR routine somehow which should notify all Virtio queues
647 * on the interrupt.
648 */
649 mutex_exit(&g_IrqMtx);
650
651 return fOurIRQ ? DDI_INTR_CLAIMED : DDI_INTR_UNCLAIMED;
652}
653
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette