VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuest/VBoxGuest-solaris.c@ 24323

Last change on this file since 24323 was 21170, checked in by vboxsync, 16 years ago

Additions: Mouse polling support in Solaris kernel module and minor common code changes.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 27.8 KB
Line 
1/* $Id: VBoxGuest-solaris.c 21170 2009-07-02 14:45:00Z vboxsync $ */
2/** @file
3 * VirtualBox Guest Additions Driver for Solaris.
4 */
5
6/*
7 * Copyright (C) 2007 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22
23/*******************************************************************************
24* Header Files *
25*******************************************************************************/
26#include <sys/conf.h>
27#include <sys/modctl.h>
28#include <sys/mutex.h>
29#include <sys/pci.h>
30#include <sys/stat.h>
31#include <sys/ddi.h>
32#include <sys/ddi_intr.h>
33#include <sys/sunddi.h>
34#include <sys/open.h>
35#undef u /* /usr/include/sys/user.h:249:1 is where this is defined to (curproc->p_user). very cool. */
36
37#include "VBoxGuestInternal.h"
38#include <VBox/log.h>
39#include <VBox/version.h>
40#include <iprt/assert.h>
41#include <iprt/initterm.h>
42#include <iprt/process.h>
43#include <iprt/mem.h>
44#include <iprt/cdefs.h>
45#include <iprt/asm.h>
46
47
48/*******************************************************************************
49* Defined Constants And Macros *
50*******************************************************************************/
51/** The module name. */
52#define DEVICE_NAME "vboxguest"
53/** The module description as seen in 'modinfo'. */
54#define DEVICE_DESC "VirtualBox GstDrv"
55
56
57/*******************************************************************************
58* Internal Functions *
59*******************************************************************************/
60static int VBoxGuestSolarisOpen(dev_t *pDev, int fFlag, int fType, cred_t *pCred);
61static int VBoxGuestSolarisClose(dev_t Dev, int fFlag, int fType, cred_t *pCred);
62static int VBoxGuestSolarisRead(dev_t Dev, struct uio *pUio, cred_t *pCred);
63static int VBoxGuestSolarisWrite(dev_t Dev, struct uio *pUio, cred_t *pCred);
64static int VBoxGuestSolarisIOCtl(dev_t Dev, int Cmd, intptr_t pArg, int Mode, cred_t *pCred, int *pVal);
65static int VBoxGuestSolarisPoll(dev_t Dev, short fEvents, int fAnyYet, short *pReqEvents, struct pollhead **ppPollHead);
66
67static int VBoxGuestSolarisGetInfo(dev_info_t *pDip, ddi_info_cmd_t enmCmd, void *pArg, void **ppResult);
68static int VBoxGuestSolarisAttach(dev_info_t *pDip, ddi_attach_cmd_t enmCmd);
69static int VBoxGuestSolarisDetach(dev_info_t *pDip, ddi_detach_cmd_t enmCmd);
70
71static int VBoxGuestSolarisAddIRQ(dev_info_t *pDip);
72static void VBoxGuestSolarisRemoveIRQ(dev_info_t *pDip);
73static uint_t VBoxGuestSolarisISR(caddr_t Arg);
74
75
76/*******************************************************************************
77* Structures and Typedefs *
78*******************************************************************************/
79/**
80 * cb_ops: for drivers that support char/block entry points
81 */
82static struct cb_ops g_VBoxGuestSolarisCbOps =
83{
84 VBoxGuestSolarisOpen,
85 VBoxGuestSolarisClose,
86 nodev, /* b strategy */
87 nodev, /* b dump */
88 nodev, /* b print */
89 VBoxGuestSolarisRead,
90 VBoxGuestSolarisWrite,
91 VBoxGuestSolarisIOCtl,
92 nodev, /* c devmap */
93 nodev, /* c mmap */
94 nodev, /* c segmap */
95 VBoxGuestSolarisPoll,
96 ddi_prop_op, /* property ops */
97 NULL, /* streamtab */
98 D_NEW | D_MP, /* compat. flag */
99 CB_REV /* revision */
100};
101
102/**
103 * dev_ops: for driver device operations
104 */
105static struct dev_ops g_VBoxGuestSolarisDevOps =
106{
107 DEVO_REV, /* driver build revision */
108 0, /* ref count */
109 VBoxGuestSolarisGetInfo,
110 nulldev, /* identify */
111 nulldev, /* probe */
112 VBoxGuestSolarisAttach,
113 VBoxGuestSolarisDetach,
114 nodev, /* reset */
115 &g_VBoxGuestSolarisCbOps,
116 (struct bus_ops *)0,
117 nodev /* power */
118};
119
120/**
121 * modldrv: export driver specifics to the kernel
122 */
123static struct modldrv g_VBoxGuestSolarisModule =
124{
125 &mod_driverops, /* extern from kernel */
126 DEVICE_DESC " " VBOX_VERSION_STRING "r" RT_XSTR(VBOX_SVN_REV),
127 &g_VBoxGuestSolarisDevOps
128};
129
130/**
131 * modlinkage: export install/remove/info to the kernel
132 */
133static struct modlinkage g_VBoxGuestSolarisModLinkage =
134{
135 MODREV_1, /* loadable module system revision */
136 &g_VBoxGuestSolarisModule,
137 NULL /* terminate array of linkage structures */
138};
139
140/**
141 * State info for each open file handle.
142 */
143typedef struct
144{
145 /** Pointer to the session handle. */
146 PVBOXGUESTSESSION pSession;
147 /** The process reference for posting signals */
148 void *pvProcRef;
149} vboxguest_state_t;
150
151
152/*******************************************************************************
153* Global Variables *
154*******************************************************************************/
155/** Device handle (we support only one instance). */
156static dev_info_t *g_pDip = NULL;
157/** Opaque pointer to file-descriptor states */
158static void *g_pVBoxGuestSolarisState = NULL;
159/** Device extention & session data association structure. */
160static VBOXGUESTDEVEXT g_DevExt;
161/** IO port handle. */
162static ddi_acc_handle_t g_PciIOHandle;
163/** MMIO handle. */
164static ddi_acc_handle_t g_PciMMIOHandle;
165/** IO Port. */
166static uint16_t g_uIOPortBase;
167/** Address of the MMIO region.*/
168static caddr_t g_pMMIOBase;
169/** Size of the MMIO region. */
170static off_t g_cbMMIO;
171/** VMMDev Version. */
172static uint32_t g_u32Version;
173/** Pointer to the interrupt handle vector */
174static ddi_intr_handle_t *g_pIntr;
175/** Number of actually allocated interrupt handles */
176static size_t g_cIntrAllocated;
177/** The pollhead structure */
178static pollhead_t g_PollHead;
179/** The IRQ Mutex */
180static kmutex_t g_IrqMtx;
181
182/**
183 * Kernel entry points
184 */
185int _init(void)
186{
187 LogFlow((DEVICE_NAME ":_init\n"));
188 int rc = ddi_soft_state_init(&g_pVBoxGuestSolarisState, sizeof(vboxguest_state_t), 1);
189 if (!rc)
190 {
191 rc = mod_install(&g_VBoxGuestSolarisModLinkage);
192 if (rc)
193 ddi_soft_state_fini(&g_pVBoxGuestSolarisState);
194 }
195 return rc;
196}
197
198
199int _fini(void)
200{
201 LogFlow((DEVICE_NAME ":_fini\n"));
202 int rc = mod_remove(&g_VBoxGuestSolarisModLinkage);
203 if (!rc)
204 ddi_soft_state_fini(&g_pVBoxGuestSolarisState);
205 return rc;
206}
207
208
209int _info(struct modinfo *pModInfo)
210{
211 LogFlow((DEVICE_NAME ":_info\n"));
212 return mod_info(&g_VBoxGuestSolarisModLinkage, pModInfo);
213}
214
215
216/**
217 * Attach entry point, to attach a device to the system or resume it.
218 *
219 * @param pDip The module structure instance.
220 * @param enmCmd Attach type (ddi_attach_cmd_t)
221 *
222 * @return corresponding solaris error code.
223 */
224static int VBoxGuestSolarisAttach(dev_info_t *pDip, ddi_attach_cmd_t enmCmd)
225{
226 LogFlow((DEVICE_NAME "::Attach\n"));
227 switch (enmCmd)
228 {
229 case DDI_ATTACH:
230 {
231 if (g_pDip)
232 {
233 LogRel((DEVICE_NAME "::Attach: Only one instance supported.\n"));
234 return DDI_FAILURE;
235 }
236
237 int rc;
238 int instance;
239
240 instance = ddi_get_instance(pDip);
241
242 /*
243 * Initialize IPRT R0 driver, which internally calls OS-specific r0 init.
244 */
245 rc = RTR0Init(0);
246 if (RT_FAILURE(rc))
247 {
248 Log((DEVICE_NAME "::Attach: RTR0Init failed.\n"));
249 return DDI_FAILURE;
250 }
251
252 /*
253 * Enable resources for PCI access.
254 */
255 ddi_acc_handle_t PciHandle;
256 rc = pci_config_setup(pDip, &PciHandle);
257 if (rc == DDI_SUCCESS)
258 {
259 /*
260 * Map the register address space.
261 */
262 caddr_t baseAddr;
263 ddi_device_acc_attr_t deviceAttr;
264 deviceAttr.devacc_attr_version = DDI_DEVICE_ATTR_V0;
265 deviceAttr.devacc_attr_endian_flags = DDI_NEVERSWAP_ACC;
266 deviceAttr.devacc_attr_dataorder = DDI_STRICTORDER_ACC;
267 deviceAttr.devacc_attr_access = DDI_DEFAULT_ACC;
268 rc = ddi_regs_map_setup(pDip, 1, &baseAddr, 0, 0, &deviceAttr, &g_PciIOHandle);
269 if (rc == DDI_SUCCESS)
270 {
271 /*
272 * Read size of the MMIO region.
273 */
274 g_uIOPortBase = (uintptr_t)baseAddr;
275 rc = ddi_dev_regsize(pDip, 2, &g_cbMMIO);
276 if (rc == DDI_SUCCESS)
277 {
278 rc = ddi_regs_map_setup(pDip, 2, &g_pMMIOBase, 0, g_cbMMIO, &deviceAttr,
279 &g_PciMMIOHandle);
280 if (rc == DDI_SUCCESS)
281 {
282 /*
283 * Add IRQ of VMMDev.
284 */
285 rc = VBoxGuestSolarisAddIRQ(pDip);
286 if (rc == DDI_SUCCESS)
287 {
288 /*
289 * Call the common device extension initializer.
290 */
291 rc = VBoxGuestInitDevExt(&g_DevExt, g_uIOPortBase, g_pMMIOBase,
292 g_cbMMIO, VBOXOSTYPE_Solaris,
293 VMMDEV_EVENT_MOUSE_POSITION_CHANGED);
294 if (RT_SUCCESS(rc))
295 {
296 rc = ddi_create_minor_node(pDip, DEVICE_NAME, S_IFCHR, instance, DDI_PSEUDO, 0);
297 if (rc == DDI_SUCCESS)
298 {
299 g_pDip = pDip;
300 pci_config_teardown(&PciHandle);
301 return DDI_SUCCESS;
302 }
303
304 LogRel((DEVICE_NAME "::Attach: ddi_create_minor_node failed.\n"));
305 VBoxGuestDeleteDevExt(&g_DevExt);
306 }
307 else
308 LogRel((DEVICE_NAME "::Attach: VBoxGuestInitDevExt failed.\n"));
309 VBoxGuestSolarisRemoveIRQ(pDip);
310 }
311 else
312 LogRel((DEVICE_NAME "::Attach: VBoxGuestSolarisAddIRQ failed.\n"));
313 ddi_regs_map_free(&g_PciMMIOHandle);
314 }
315 else
316 LogRel((DEVICE_NAME "::Attach: ddi_regs_map_setup for MMIO region failed.\n"));
317 }
318 else
319 LogRel((DEVICE_NAME "::Attach: ddi_dev_regsize for MMIO region failed.\n"));
320 ddi_regs_map_free(&g_PciIOHandle);
321 }
322 else
323 LogRel((DEVICE_NAME "::Attach: ddi_regs_map_setup for IOport failed.\n"));
324 pci_config_teardown(&PciHandle);
325 }
326 else
327 LogRel((DEVICE_NAME "::Attach: pci_config_setup failed rc=%d.\n", rc));
328
329 RTR0Term();
330 return DDI_FAILURE;
331 }
332
333 case DDI_RESUME:
334 {
335 /** @todo implement resume for guest driver. */
336 return DDI_SUCCESS;
337 }
338
339 default:
340 return DDI_FAILURE;
341 }
342}
343
344
345/**
346 * Detach entry point, to detach a device to the system or suspend it.
347 *
348 * @param pDip The module structure instance.
349 * @param enmCmd Attach type (ddi_attach_cmd_t)
350 *
351 * @return corresponding solaris error code.
352 */
353static int VBoxGuestSolarisDetach(dev_info_t *pDip, ddi_detach_cmd_t enmCmd)
354{
355 LogFlow((DEVICE_NAME "::Detach\n"));
356 switch (enmCmd)
357 {
358 case DDI_DETACH:
359 {
360 VBoxGuestSolarisRemoveIRQ(pDip);
361 ddi_regs_map_free(&g_PciIOHandle);
362 ddi_regs_map_free(&g_PciMMIOHandle);
363 ddi_remove_minor_node(pDip, NULL);
364
365 RTR0Term();
366 return DDI_SUCCESS;
367 }
368
369 case DDI_SUSPEND:
370 {
371 /** @todo implement suspend for guest driver. */
372 return DDI_SUCCESS;
373 }
374
375 default:
376 return DDI_FAILURE;
377 }
378}
379
380
381/**
382 * Info entry point, called by solaris kernel for obtaining driver info.
383 *
384 * @param pDip The module structure instance (do not use).
385 * @param enmCmd Information request type.
386 * @param pvArg Type specific argument.
387 * @param ppvResult Where to store the requested info.
388 *
389 * @return corresponding solaris error code.
390 */
391static int VBoxGuestSolarisGetInfo(dev_info_t *pDip, ddi_info_cmd_t enmCmd, void *pvArg, void **ppvResult)
392{
393 LogFlow((DEVICE_NAME "::GetInfo\n"));
394
395 int rc = DDI_SUCCESS;
396 switch (enmCmd)
397 {
398 case DDI_INFO_DEVT2DEVINFO:
399 *ppvResult = (void *)g_pDip;
400 break;
401
402 case DDI_INFO_DEVT2INSTANCE:
403 *ppvResult = (void *)(uintptr_t)ddi_get_instance(g_pDip);
404 break;
405
406 default:
407 rc = DDI_FAILURE;
408 break;
409 }
410
411 NOREF(pvArg);
412 return rc;
413}
414
415
416/**
417 * User context entry points
418 */
419static int VBoxGuestSolarisOpen(dev_t *pDev, int fFlag, int fType, cred_t *pCred)
420{
421 int rc;
422 PVBOXGUESTSESSION pSession;
423
424 LogFlow((DEVICE_NAME "::Open\n"));
425
426 /*
427 * Verify we are being opened as a character device.
428 */
429 if (fType != OTYP_CHR)
430 return EINVAL;
431
432 vboxguest_state_t *pState = NULL;
433 unsigned iOpenInstance;
434 for (iOpenInstance = 0; iOpenInstance < 4096; iOpenInstance++)
435 {
436 if ( !ddi_get_soft_state(g_pVBoxGuestSolarisState, iOpenInstance) /* faster */
437 && ddi_soft_state_zalloc(g_pVBoxGuestSolarisState, iOpenInstance) == DDI_SUCCESS)
438 {
439 pState = ddi_get_soft_state(g_pVBoxGuestSolarisState, iOpenInstance);
440 break;
441 }
442 }
443 if (!pState)
444 {
445 Log((DEVICE_NAME "::Open: too many open instances."));
446 return ENXIO;
447 }
448
449 /*
450 * Create a new session.
451 */
452 rc = VBoxGuestCreateUserSession(&g_DevExt, &pSession);
453 if (RT_SUCCESS(rc))
454 {
455 pState->pvProcRef = proc_ref();
456 pState->pSession = pSession;
457 *pDev = makedevice(getmajor(*pDev), iOpenInstance);
458 Log((DEVICE_NAME "::Open: pSession=%p pState=%p pid=%d\n", pSession, pState, (int)RTProcSelf()));
459 return 0;
460 }
461
462 /* Failed, clean up. */
463 ddi_soft_state_free(g_pVBoxGuestSolarisState, iOpenInstance);
464
465 LogRel((DEVICE_NAME "::Open: VBoxGuestCreateUserSession failed. rc=%d\n", rc));
466 return EFAULT;
467}
468
469
470static int VBoxGuestSolarisClose(dev_t Dev, int flag, int fType, cred_t *pCred)
471{
472 LogFlow((DEVICE_NAME "::Close pid=%d\n", (int)RTProcSelf()));
473
474 PVBOXGUESTSESSION pSession;
475 vboxguest_state_t *pState = ddi_get_soft_state(g_pVBoxGuestSolarisState, getminor(Dev));
476 if (!pState)
477 {
478 Log((DEVICE_NAME "::Close: failed to get pState.\n"));
479 return EFAULT;
480 }
481
482 proc_unref(pState->pvProcRef);
483 pSession = pState->pSession;
484 pState->pSession = NULL;
485 Log((DEVICE_NAME "::Close: pSession=%p pState=%p\n", pSession, pState));
486 ddi_soft_state_free(g_pVBoxGuestSolarisState, getminor(Dev));
487 if (!pSession)
488 {
489 Log((DEVICE_NAME "::Close: failed to get pSession.\n"));
490 return EFAULT;
491 }
492
493 /*
494 * Close the session.
495 */
496 VBoxGuestCloseSession(&g_DevExt, pSession);
497 return 0;
498}
499
500
501static int VBoxGuestSolarisRead(dev_t Dev, struct uio *pUio, cred_t *pCred)
502{
503 LogFlow((DEVICE_NAME "::Read\n"));
504
505 PVBOXGUESTSESSION pSession;
506 vboxguest_state_t *pState = ddi_get_soft_state(g_pVBoxGuestSolarisState, getminor(Dev));
507 if (!pState)
508 {
509 Log((DEVICE_NAME "::Close: failed to get pState.\n"));
510 return EFAULT;
511 }
512
513 uint32_t u32CurSeq = ASMAtomicUoReadU32(&g_DevExt.u32MousePosChangedSeq);
514 if (pSession->u32MousePosChangedSeq != u32CurSeq)
515 pSession->u32MousePosChangedSeq = u32CurSeq;
516
517 return 0;
518}
519
520
521static int VBoxGuestSolarisWrite(dev_t Dev, struct uio *pUio, cred_t *pCred)
522{
523 LogFlow((DEVICE_NAME "::Write\n"));
524 return 0;
525}
526
527
528/** @def IOCPARM_LEN
529 * Gets the length from the ioctl number.
530 * This is normally defined by sys/ioccom.h on BSD systems...
531 */
532#ifndef IOCPARM_LEN
533# define IOCPARM_LEN(Code) (((Code) >> 16) & IOCPARM_MASK)
534#endif
535
536
537/**
538 * Driver ioctl, an alternate entry point for this character driver.
539 *
540 * @param Dev Device number
541 * @param Cmd Operation identifier
542 * @param pArg Arguments from user to driver
543 * @param Mode Information bitfield (read/write, address space etc.)
544 * @param pCred User credentials
545 * @param pVal Return value for calling process.
546 *
547 * @return corresponding solaris error code.
548 */
549static int VBoxGuestSolarisIOCtl(dev_t Dev, int Cmd, intptr_t pArg, int Mode, cred_t *pCred, int *pVal)
550{
551 LogFlow((DEVICE_NAME ":VBoxGuestSolarisIOCtl\n"));
552
553 /*
554 * Get the session from the soft state item.
555 */
556 vboxguest_state_t *pState = ddi_get_soft_state(g_pVBoxGuestSolarisState, getminor(Dev));
557 if (!pState)
558 {
559 Log((DEVICE_NAME "::IOCtl: no state data for %d\n", getminor(Dev)));
560 return EINVAL;
561 }
562
563 PVBOXGUESTSESSION pSession = pState->pSession;
564 if (!pSession)
565 {
566 Log((DEVICE_NAME "::IOCtl: no session data for %d\n", getminor(Dev)));
567 return EINVAL;
568 }
569
570 /*
571 * Read and validate the request wrapper.
572 */
573 VBGLBIGREQ ReqWrap;
574 if (IOCPARM_LEN(Cmd) != sizeof(ReqWrap))
575 {
576 LogRel((DEVICE_NAME "::IOCtl: bad request %#x size=%d expected=%d\n", Cmd, IOCPARM_LEN(Cmd), sizeof(ReqWrap)));
577 return ENOTTY;
578 }
579
580 int rc = ddi_copyin((void *)pArg, &ReqWrap, sizeof(ReqWrap), Mode);
581 if (RT_UNLIKELY(rc))
582 {
583 LogRel((DEVICE_NAME "::IOCtl: ddi_copyin failed to read header pArg=%p Cmd=%d. rc=%d.\n", pArg, Cmd, rc));
584 return EINVAL;
585 }
586
587 if (ReqWrap.u32Magic != VBGLBIGREQ_MAGIC)
588 {
589 LogRel((DEVICE_NAME "::IOCtl: bad magic %#x; pArg=%p Cmd=%d.\n", ReqWrap.u32Magic, pArg, Cmd));
590 return EINVAL;
591 }
592 if (RT_UNLIKELY( ReqWrap.cbData == 0
593 || ReqWrap.cbData > _1M*16))
594 {
595 Log((DEVICE_NAME "::IOCtl: bad size %#x; pArg=%p Cmd=%d.\n", ReqWrap.cbData, pArg, Cmd));
596 return EINVAL;
597 }
598
599 /*
600 * Read the request.
601 */
602 void *pvBuf = RTMemTmpAlloc(ReqWrap.cbData);
603 if (RT_UNLIKELY(!pvBuf))
604 {
605 LogRel((DEVICE_NAME "::IOCtl: RTMemTmpAlloc failed to alloc %d bytes.\n", ReqWrap.cbData));
606 return ENOMEM;
607 }
608
609 rc = ddi_copyin((void *)(uintptr_t)ReqWrap.pvDataR3, pvBuf, ReqWrap.cbData, Mode);
610 if (RT_UNLIKELY(rc))
611 {
612 RTMemTmpFree(pvBuf);
613 LogRel((DEVICE_NAME "::IOCtl: ddi_copyin failed; pvBuf=%p pArg=%p Cmd=%d. rc=%d\n", pvBuf, pArg, Cmd, rc));
614 return EFAULT;
615 }
616 if (RT_UNLIKELY( ReqWrap.cbData != 0
617 && !VALID_PTR(pvBuf)))
618 {
619 RTMemTmpFree(pvBuf);
620 LogRel((DEVICE_NAME "::IOCtl: pvBuf invalid pointer %p\n", pvBuf));
621 return EINVAL;
622 }
623 Log((DEVICE_NAME "::IOCtl: pSession=%p pid=%d.\n", pSession, (int)RTProcSelf()));
624
625 /*
626 * Process the IOCtl.
627 */
628 size_t cbDataReturned;
629 rc = VBoxGuestCommonIOCtl(Cmd, &g_DevExt, pSession, pvBuf, ReqWrap.cbData, &cbDataReturned);
630 if (RT_SUCCESS(rc))
631 {
632 rc = 0;
633 if (RT_UNLIKELY(cbDataReturned > ReqWrap.cbData))
634 {
635 LogRel((DEVICE_NAME "::IOCtl: too much output data %d expected %d\n", cbDataReturned, ReqWrap.cbData));
636 cbDataReturned = ReqWrap.cbData;
637 }
638 if (cbDataReturned > 0)
639 {
640 rc = ddi_copyout(pvBuf, (void *)(uintptr_t)ReqWrap.pvDataR3, cbDataReturned, Mode);
641 if (RT_UNLIKELY(rc))
642 {
643 LogRel((DEVICE_NAME "::IOCtl: ddi_copyout failed; pvBuf=%p pArg=%p cbDataReturned=%u Cmd=%d. rc=%d\n",
644 pvBuf, pArg, cbDataReturned, Cmd, rc));
645 rc = EFAULT;
646 }
647 }
648 }
649 else
650 {
651 LogRel((DEVICE_NAME "::IOCtl: VBoxGuestCommonIOCtl failed. rc=%d\n", rc));
652 rc = RTErrConvertToErrno(rc);
653 }
654 *pVal = rc;
655 RTMemTmpFree(pvBuf);
656 return rc;
657}
658
659
660static int VBoxGuestSolarisPoll(dev_t Dev, short fEvents, int fAnyYet, short *pReqEvents, struct pollhead **ppPollHead)
661{
662 LogFlow((DEVICE_NAME "::Poll: fEvents=%d fAnyYet=%d\n", fEvents, fAnyYet));
663
664 vboxguest_state_t *pState = ddi_get_soft_state(g_pVBoxGuestSolarisState, getminor(Dev));
665 if (RT_LIKELY(pState))
666 {
667 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pState->pSession;
668 uint32_t u32CurSeq = ASMAtomicUoReadU32(&g_DevExt.u32MousePosChangedSeq);
669 if (pSession->u32MousePosChangedSeq != u32CurSeq)
670 {
671 *pReqEvents |= (POLLIN | POLLRDNORM);
672 pSession->u32MousePosChangedSeq = u32CurSeq;
673 }
674 else
675 {
676 *pReqEvents = 0;
677 if (!fAnyYet)
678 *ppPollHead = &g_PollHead;
679 }
680
681 return 0;
682 }
683 else
684 {
685 Log((DEVICE_NAME "::Poll: no state data for %d\n", getminor(Dev)));
686 return EINVAL;
687 }
688}
689
690
691/**
692 * Sets IRQ for VMMDev.
693 *
694 * @returns Solaris error code.
695 * @param pDip Pointer to the device info structure.
696 */
697static int VBoxGuestSolarisAddIRQ(dev_info_t *pDip)
698{
699 LogFlow((DEVICE_NAME "::AddIRQ: pDip=%p\n", pDip));
700
701 int IntrType = 0;
702 int rc = ddi_intr_get_supported_types(pDip, &IntrType);
703 if (rc == DDI_SUCCESS)
704 {
705 /* We won't need to bother about MSIs. */
706 if (IntrType & DDI_INTR_TYPE_FIXED)
707 {
708 int IntrCount = 0;
709 rc = ddi_intr_get_nintrs(pDip, IntrType, &IntrCount);
710 if ( rc == DDI_SUCCESS
711 && IntrCount > 0)
712 {
713 int IntrAvail = 0;
714 rc = ddi_intr_get_navail(pDip, IntrType, &IntrAvail);
715 if ( rc == DDI_SUCCESS
716 && IntrAvail > 0)
717 {
718 /* Allocated kernel memory for the interrupt handles. The allocation size is stored internally. */
719 g_pIntr = RTMemAlloc(IntrCount * sizeof(ddi_intr_handle_t));
720 if (g_pIntr)
721 {
722 int IntrAllocated;
723 rc = ddi_intr_alloc(pDip, g_pIntr, IntrType, 0, IntrCount, &IntrAllocated, DDI_INTR_ALLOC_NORMAL);
724 if ( rc == DDI_SUCCESS
725 && IntrAllocated > 0)
726 {
727 g_cIntrAllocated = IntrAllocated;
728 uint_t uIntrPriority;
729 rc = ddi_intr_get_pri(g_pIntr[0], &uIntrPriority);
730 if (rc == DDI_SUCCESS)
731 {
732 /* Initialize the mutex. */
733 mutex_init(&g_IrqMtx, NULL, MUTEX_DRIVER, DDI_INTR_PRI(uIntrPriority));
734
735 /* Assign interrupt handler functions and enable interrupts. */
736 for (int i = 0; i < IntrAllocated; i++)
737 {
738 rc = ddi_intr_add_handler(g_pIntr[i], (ddi_intr_handler_t *)VBoxGuestSolarisISR,
739 NULL /* No Private Data */, NULL);
740 if (rc == DDI_SUCCESS)
741 rc = ddi_intr_enable(g_pIntr[i]);
742 if (rc != DDI_SUCCESS)
743 {
744 /* Changing local IntrAllocated to hold so-far allocated handles for freeing. */
745 IntrAllocated = i;
746 break;
747 }
748 }
749 if (rc == DDI_SUCCESS)
750 return rc;
751
752 /* Remove any assigned handlers */
753 LogRel((DEVICE_NAME ":failed to assign IRQs allocated=%d\n", IntrAllocated));
754 for (int x = 0; x < IntrAllocated; x++)
755 ddi_intr_remove_handler(g_pIntr[x]);
756 }
757 else
758 LogRel((DEVICE_NAME "::AddIRQ: failed to get priority of interrupt. rc=%d\n", rc));
759
760 /* Remove allocated IRQs, too bad we can free only one handle at a time. */
761 for (int k = 0; k < g_cIntrAllocated; k++)
762 ddi_intr_free(g_pIntr[k]);
763 }
764 else
765 LogRel((DEVICE_NAME "::AddIRQ: failed to allocated IRQs. count=%d\n", IntrCount));
766 RTMemFree(g_pIntr);
767 }
768 else
769 LogRel((DEVICE_NAME "::AddIRQ: failed to allocated IRQs. count=%d\n", IntrCount));
770 }
771 else
772 LogRel((DEVICE_NAME "::AddIRQ: failed to get or insufficient available IRQs. rc=%d IntrAvail=%d\n", rc, IntrAvail));
773 }
774 else
775 LogRel((DEVICE_NAME "::AddIRQ: failed to get or insufficient number of IRQs. rc=%d IntrCount=%d\n", rc, IntrCount));
776 }
777 else
778 LogRel((DEVICE_NAME "::AddIRQ: invalid irq type. IntrType=%#x\n", IntrType));
779 }
780 else
781 LogRel((DEVICE_NAME "::AddIRQ: failed to get supported interrupt types\n"));
782 return rc;
783}
784
785
786/**
787 * Removes IRQ for VMMDev.
788 *
789 * @param pDip Pointer to the device info structure.
790 */
791static void VBoxGuestSolarisRemoveIRQ(dev_info_t *pDip)
792{
793 LogFlow((DEVICE_NAME "::RemoveIRQ:\n"));
794
795 for (int i = 0; i < g_cIntrAllocated; i++)
796 {
797 int rc = ddi_intr_disable(g_pIntr[i]);
798 if (rc == DDI_SUCCESS)
799 {
800 rc = ddi_intr_remove_handler(g_pIntr[i]);
801 if (rc == DDI_SUCCESS)
802 ddi_intr_free(g_pIntr[i]);
803 }
804 }
805 RTMemFree(g_pIntr);
806 mutex_destroy(&g_IrqMtx);
807}
808
809
810/**
811 * Interrupt Service Routine for VMMDev.
812 *
813 * @param Arg Private data (unused, will be NULL).
814 * @returns DDI_INTR_CLAIMED if it's our interrupt, DDI_INTR_UNCLAIMED if it isn't.
815 */
816static uint_t VBoxGuestSolarisISR(caddr_t Arg)
817{
818 LogFlow((DEVICE_NAME "::ISR:\n"));
819
820 mutex_enter(&g_IrqMtx);
821 bool fOurIRQ = VBoxGuestCommonISR(&g_DevExt);
822 mutex_exit(&g_IrqMtx);
823
824 return fOurIRQ ? DDI_INTR_CLAIMED : DDI_INTR_UNCLAIMED;
825}
826
827
828void VBoxGuestNativeISRMousePollEvent(PVBOXGUESTDEVEXT pDevExt)
829{
830 LogFlow((DEVICE_NAME "::NativeISRMousePollEvent:\n"));
831
832 /*
833 * Wake up poll waiters.
834 */
835 pollwakeup(&g_PollHead, POLLIN | POLLRDNORM);
836}
837
838
839/* Common code that depend on g_DevExt. */
840#include "VBoxGuestIDC-unix.c.h"
841
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