VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuest/VBoxGuest-darwin.cpp@ 82853

Last change on this file since 82853 was 76553, checked in by vboxsync, 6 years ago

scm --update-copyright-year

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 44.7 KB
Line 
1/* $Id: VBoxGuest-darwin.cpp 76553 2019-01-01 01:45:53Z vboxsync $ */
2/** @file
3 * VBoxGuest - Darwin Specifics.
4 */
5
6/*
7 * Copyright (C) 2006-2019 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/*********************************************************************************************************************************
29* Header Files *
30*********************************************************************************************************************************/
31#define LOG_GROUP LOG_GROUP_VGDRV
32/*
33 * Deal with conflicts first.
34 * PVM - BSD mess, that FreeBSD has correct a long time ago.
35 * iprt/types.h before sys/param.h - prevents UINT32_C and friends.
36 */
37#include <iprt/types.h>
38#include <sys/param.h>
39#undef PVM
40
41#include <IOKit/IOLib.h> /* Assert as function */
42
43#include <VBox/version.h>
44#include <iprt/asm.h>
45#include <iprt/assert.h>
46#include <iprt/initterm.h>
47#include <iprt/mem.h>
48#include <iprt/power.h>
49#include <iprt/process.h>
50#include <iprt/semaphore.h>
51#include <iprt/spinlock.h>
52#include <iprt/string.h>
53#include <VBox/err.h>
54#include <VBox/log.h>
55
56#include <mach/kmod.h>
57#include <miscfs/devfs/devfs.h>
58#include <sys/conf.h>
59#include <sys/errno.h>
60#include <sys/ioccom.h>
61#include <sys/malloc.h>
62#include <sys/proc.h>
63#include <sys/kauth.h>
64#include <IOKit/IOService.h>
65#include <IOKit/IOUserClient.h>
66#include <IOKit/pwr_mgt/RootDomain.h>
67#include <IOKit/pci/IOPCIDevice.h>
68#include <IOKit/IOBufferMemoryDescriptor.h>
69#include <IOKit/IOFilterInterruptEventSource.h>
70#include "VBoxGuestInternal.h"
71
72
73/*********************************************************************************************************************************
74* Defined Constants And Macros *
75*********************************************************************************************************************************/
76/** The system device node name. */
77#define DEVICE_NAME_SYS "vboxguest"
78/** The user device node name. */
79#define DEVICE_NAME_USR "vboxguestu"
80
81
82/** @name For debugging/whatever, now permanent.
83 * @{ */
84#define VBOX_PROC_SELFNAME_LEN 31
85#define VBOX_RETRIEVE_CUR_PROC_NAME(a_Name) char a_Name[VBOX_PROC_SELFNAME_LEN + 1]; \
86 proc_selfname(a_Name, VBOX_PROC_SELFNAME_LEN)
87/** @} */
88
89
90/*********************************************************************************************************************************
91* Internal Functions *
92*********************************************************************************************************************************/
93RT_C_DECLS_BEGIN
94static kern_return_t vgdrvDarwinStart(struct kmod_info *pKModInfo, void *pvData);
95static kern_return_t vgdrvDarwinStop(struct kmod_info *pKModInfo, void *pvData);
96static int vgdrvDarwinCharDevRemove(void);
97
98static int vgdrvDarwinOpen(dev_t Dev, int fFlags, int fDevType, struct proc *pProcess);
99static int vgdrvDarwinClose(dev_t Dev, int fFlags, int fDevType, struct proc *pProcess);
100static int vgdrvDarwinIOCtlSlow(PVBOXGUESTSESSION pSession, u_long iCmd, caddr_t pData, struct proc *pProcess);
101static int vgdrvDarwinIOCtl(dev_t Dev, u_long iCmd, caddr_t pData, int fFlags, struct proc *pProcess);
102
103static int vgdrvDarwinErr2DarwinErr(int rc);
104
105static IOReturn vgdrvDarwinSleepHandler(void *pvTarget, void *pvRefCon, UInt32 uMessageType, IOService *pProvider, void *pvMessageArgument, vm_size_t argSize);
106RT_C_DECLS_END
107
108
109/*********************************************************************************************************************************
110* Structures and Typedefs *
111*********************************************************************************************************************************/
112/**
113 * The service class for handling the VMMDev PCI device.
114 *
115 * Instantiated when the module is loaded (and on PCI hotplugging?).
116 */
117class org_virtualbox_VBoxGuest : public IOService
118{
119 OSDeclareDefaultStructors(org_virtualbox_VBoxGuest);
120
121private:
122 IOPCIDevice *m_pIOPCIDevice;
123 IOMemoryMap *m_pMap;
124 IOFilterInterruptEventSource *m_pInterruptSrc;
125
126 bool setupVmmDevInterrupts(IOService *pProvider);
127 bool disableVmmDevInterrupts(void);
128 bool isVmmDev(IOPCIDevice *pIOPCIDevice);
129
130protected:
131 /** Non-NULL if interrupts are registered. Probably same as getProvider(). */
132 IOService *m_pInterruptProvider;
133
134public:
135 virtual bool init(OSDictionary *pDictionary = 0);
136 virtual void free(void);
137 virtual IOService *probe(IOService *pProvider, SInt32 *pi32Score);
138 virtual bool start(IOService *pProvider);
139 virtual void stop(IOService *pProvider);
140 virtual bool terminate(IOOptionBits fOptions);
141 static void vgdrvDarwinIrqHandler(OSObject *pTarget, void *pvRefCon, IOService *pNub, int iSrc);
142};
143
144OSDefineMetaClassAndStructors(org_virtualbox_VBoxGuest, IOService);
145
146
147/**
148 * An attempt at getting that clientDied() notification.
149 * I don't think it'll work as I cannot figure out where/what creates the correct
150 * port right.
151 *
152 * Instantiated when userland does IOServiceOpen().
153 */
154class org_virtualbox_VBoxGuestClient : public IOUserClient
155{
156 OSDeclareDefaultStructors(org_virtualbox_VBoxGuestClient);
157
158private:
159 /** Guard against the parent class growing and us using outdated headers. */
160 uint8_t m_abSafetyPadding[256];
161
162 PVBOXGUESTSESSION m_pSession; /**< The session. */
163 task_t m_Task; /**< The client task. */
164 org_virtualbox_VBoxGuest *m_pProvider; /**< The service provider. */
165
166public:
167 virtual bool initWithTask(task_t OwningTask, void *pvSecurityId, UInt32 u32Type);
168 virtual bool start(IOService *pProvider);
169 static void sessionClose(RTPROCESS Process);
170 virtual IOReturn clientClose(void);
171 virtual IOReturn clientDied(void);
172 virtual bool terminate(IOOptionBits fOptions = 0);
173 virtual bool finalize(IOOptionBits fOptions);
174 virtual void stop(IOService *pProvider);
175
176 RTR0MEMEF_NEW_AND_DELETE_OPERATORS_IOKIT();
177};
178
179OSDefineMetaClassAndStructors(org_virtualbox_VBoxGuestClient, IOUserClient);
180
181
182
183/*********************************************************************************************************************************
184* Global Variables *
185*********************************************************************************************************************************/
186/**
187 * Declare the module stuff.
188 */
189RT_C_DECLS_BEGIN
190extern kern_return_t _start(struct kmod_info *pKModInfo, void *pvData);
191extern kern_return_t _stop(struct kmod_info *pKModInfo, void *pvData);
192
193KMOD_EXPLICIT_DECL(VBoxGuest, VBOX_VERSION_STRING, _start, _stop)
194DECLHIDDEN(kmod_start_func_t *) _realmain = vgdrvDarwinStart;
195DECLHIDDEN(kmod_stop_func_t *) _antimain = vgdrvDarwinStop;
196DECLHIDDEN(int) _kext_apple_cc = __APPLE_CC__;
197RT_C_DECLS_END
198
199
200/**
201 * Device extention & session data association structure.
202 */
203static VBOXGUESTDEVEXT g_DevExt;
204
205/**
206 * The character device switch table for the driver.
207 */
208static struct cdevsw g_DevCW =
209{
210 /*.d_open = */ vgdrvDarwinOpen,
211 /*.d_close = */ vgdrvDarwinClose,
212 /*.d_read = */ eno_rdwrt,
213 /*.d_write = */ eno_rdwrt,
214 /*.d_ioctl = */ vgdrvDarwinIOCtl,
215 /*.d_stop = */ eno_stop,
216 /*.d_reset = */ eno_reset,
217 /*.d_ttys = */ NULL,
218 /*.d_select = */ eno_select,
219 /*.d_mmap = */ eno_mmap,
220 /*.d_strategy = */ eno_strat,
221 /*.d_getc = */ (void *)(uintptr_t)&enodev, //eno_getc,
222 /*.d_putc = */ (void *)(uintptr_t)&enodev, //eno_putc,
223 /*.d_type = */ 0
224};
225
226/** Major device number. */
227static int g_iMajorDeviceNo = -1;
228/** Registered devfs device handle. */
229static void *g_hDevFsDeviceSys = NULL;
230/** Registered devfs device handle for the user device. */
231static void *g_hDevFsDeviceUsr = NULL; /**< @todo 4 later */
232
233/** Spinlock protecting g_apSessionHashTab. */
234static RTSPINLOCK g_Spinlock = NIL_RTSPINLOCK;
235/** Hash table */
236static PVBOXGUESTSESSION g_apSessionHashTab[19];
237/** Calculates the index into g_apSessionHashTab.*/
238#define SESSION_HASH(pid) ((pid) % RT_ELEMENTS(g_apSessionHashTab))
239/** The number of open sessions. */
240static int32_t volatile g_cSessions = 0;
241/** Makes sure there is only one org_virtualbox_VBoxGuest instance. */
242static bool volatile g_fInstantiated = 0;
243/** The notifier handle for the sleep callback handler. */
244static IONotifier *g_pSleepNotifier = NULL;
245
246
247/**
248 * Start the kernel module.
249 */
250static kern_return_t vgdrvDarwinStart(struct kmod_info *pKModInfo, void *pvData)
251{
252 RT_NOREF(pKModInfo, pvData);
253#ifdef DEBUG
254 printf("vgdrvDarwinStart\n");
255#endif
256#if 0
257 gIOKitDebug |= 0x001 //kIOLogAttach
258 | 0x002 //kIOLogProbe
259 | 0x004 //kIOLogStart
260 | 0x008 //kIOLogRegister
261 | 0x010 //kIOLogMatch
262 | 0x020 //kIOLogConfig
263 ;
264#endif
265
266 /*
267 * Initialize IPRT.
268 */
269 int rc = RTR0Init(0);
270 if (RT_SUCCESS(rc))
271 {
272 Log(("VBoxGuest: driver loaded\n"));
273 return KMOD_RETURN_SUCCESS;
274 }
275
276 RTLogBackdoorPrintf("VBoxGuest: RTR0Init failed with rc=%Rrc\n", rc);
277 printf("VBoxGuest: RTR0Init failed with rc=%d\n", rc);
278 return KMOD_RETURN_FAILURE;
279}
280
281
282/**
283 * Stop the kernel module.
284 */
285static kern_return_t vgdrvDarwinStop(struct kmod_info *pKModInfo, void *pvData)
286{
287 RT_NOREF(pKModInfo, pvData);
288
289 /** @todo we need to check for VBoxSF clients? */
290
291 RTLogBackdoorPrintf("VBoxGuest: calling RTR0TermForced ...\n");
292 RTR0TermForced();
293
294 RTLogBackdoorPrintf("VBoxGuest: vgdrvDarwinStop returns.\n");
295 printf("VBoxGuest: driver unloaded\n");
296 return KMOD_RETURN_SUCCESS;
297}
298
299
300/**
301 * Register VBoxGuest char device
302 */
303static int vgdrvDarwinCharDevInit(void)
304{
305 int rc = RTSpinlockCreate(&g_Spinlock, RTSPINLOCK_FLAGS_INTERRUPT_SAFE, "VBoxGuestDarwin");
306 if (RT_SUCCESS(rc))
307 {
308 /*
309 * Registering ourselves as a character device.
310 */
311 g_iMajorDeviceNo = cdevsw_add(-1, &g_DevCW);
312 if (g_iMajorDeviceNo >= 0)
313 {
314 /** @todo limit /dev/vboxguest access. */
315 g_hDevFsDeviceSys = devfs_make_node(makedev(g_iMajorDeviceNo, 0), DEVFS_CHAR,
316 UID_ROOT, GID_WHEEL, 0666, DEVICE_NAME_SYS);
317 if (g_hDevFsDeviceSys != NULL)
318 {
319 /*
320 * And a all-user device.
321 */
322 g_hDevFsDeviceUsr = devfs_make_node(makedev(g_iMajorDeviceNo, 1), DEVFS_CHAR,
323 UID_ROOT, GID_WHEEL, 0666, DEVICE_NAME_USR);
324 if (g_hDevFsDeviceUsr != NULL)
325 {
326 /*
327 * Register a sleep/wakeup notification callback.
328 */
329 g_pSleepNotifier = registerPrioritySleepWakeInterest(&vgdrvDarwinSleepHandler, &g_DevExt, NULL);
330 if (g_pSleepNotifier != NULL)
331 return KMOD_RETURN_SUCCESS;
332 }
333 }
334 }
335 vgdrvDarwinCharDevRemove();
336 }
337 return KMOD_RETURN_FAILURE;
338}
339
340
341/**
342 * Unregister VBoxGuest char devices and associated session spinlock.
343 */
344static int vgdrvDarwinCharDevRemove(void)
345{
346 if (g_pSleepNotifier)
347 {
348 g_pSleepNotifier->remove();
349 g_pSleepNotifier = NULL;
350 }
351
352 if (g_hDevFsDeviceSys)
353 {
354 devfs_remove(g_hDevFsDeviceSys);
355 g_hDevFsDeviceSys = NULL;
356 }
357
358 if (g_hDevFsDeviceUsr)
359 {
360 devfs_remove(g_hDevFsDeviceUsr);
361 g_hDevFsDeviceUsr = NULL;
362 }
363
364 if (g_iMajorDeviceNo != -1)
365 {
366 int rc2 = cdevsw_remove(g_iMajorDeviceNo, &g_DevCW);
367 Assert(rc2 == g_iMajorDeviceNo); NOREF(rc2);
368 g_iMajorDeviceNo = -1;
369 }
370
371 if (g_Spinlock != NIL_RTSPINLOCK)
372 {
373 int rc2 = RTSpinlockDestroy(g_Spinlock); AssertRC(rc2);
374 g_Spinlock = NIL_RTSPINLOCK;
375 }
376
377 return KMOD_RETURN_SUCCESS;
378}
379
380
381/**
382 * Device open. Called on open /dev/vboxguest and (later) /dev/vboxguestu.
383 *
384 * @param Dev The device number.
385 * @param fFlags ???.
386 * @param fDevType ???.
387 * @param pProcess The process issuing this request.
388 */
389static int vgdrvDarwinOpen(dev_t Dev, int fFlags, int fDevType, struct proc *pProcess)
390{
391 RT_NOREF(fFlags, fDevType);
392
393 /*
394 * Only two minor devices numbers are allowed.
395 */
396 if (minor(Dev) != 0 && minor(Dev) != 1)
397 return EACCES;
398
399 /*
400 * The process issuing the request must be the current process.
401 */
402 RTPROCESS Process = RTProcSelf();
403 if ((int)Process != proc_pid(pProcess))
404 return EIO;
405
406 /*
407 * Find the session created by org_virtualbox_VBoxGuestClient, fail
408 * if no such session, and mark it as opened. We set the uid & gid
409 * here too, since that is more straight forward at this point.
410 */
411 const bool fUnrestricted = minor(Dev) == 0;
412 int rc = VINF_SUCCESS;
413 PVBOXGUESTSESSION pSession = NULL;
414 kauth_cred_t pCred = kauth_cred_proc_ref(pProcess);
415 if (pCred)
416 {
417#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1070
418 RTUID Uid = kauth_cred_getruid(pCred);
419 RTGID Gid = kauth_cred_getrgid(pCred);
420#else
421 RTUID Uid = pCred->cr_ruid;
422 RTGID Gid = pCred->cr_rgid;
423#endif
424 unsigned iHash = SESSION_HASH(Process);
425 RTSpinlockAcquire(g_Spinlock);
426
427 pSession = g_apSessionHashTab[iHash];
428 while (pSession && pSession->Process != Process)
429 pSession = pSession->pNextHash;
430 if (pSession)
431 {
432 if (!pSession->fOpened)
433 {
434 pSession->fOpened = true;
435 pSession->fUserSession = !fUnrestricted;
436 pSession->fRequestor = VMMDEV_REQUESTOR_USERMODE | VMMDEV_REQUESTOR_TRUST_NOT_GIVEN;
437 if (Uid == 0)
438 pSession->fRequestor |= VMMDEV_REQUESTOR_USR_ROOT;
439 else
440 pSession->fRequestor |= VMMDEV_REQUESTOR_USR_USER;
441 if (Gid == 0)
442 pSession->fRequestor |= VMMDEV_REQUESTOR_GRP_WHEEL;
443 if (!fUnrestricted)
444 pSession->fRequestor |= VMMDEV_REQUESTOR_USER_DEVICE;
445 pSession->fRequestor |= VMMDEV_REQUESTOR_CON_DONT_KNOW; /** @todo see if we can figure out console relationship of pProc. */
446 }
447 else
448 rc = VERR_ALREADY_LOADED;
449 }
450 else
451 rc = VERR_GENERAL_FAILURE;
452
453 RTSpinlockRelease(g_Spinlock);
454#if MAC_OS_X_VERSION_MIN_REQUIRED >= 1050
455 kauth_cred_unref(&pCred);
456#else /* 10.4 */
457 /* The 10.4u SDK headers and 10.4.11 kernel source have inconsistent definitions
458 of kauth_cred_unref(), so use the other (now deprecated) API for releasing it. */
459 kauth_cred_rele(pCred);
460#endif /* 10.4 */
461 }
462 else
463 rc = VERR_INVALID_PARAMETER;
464
465 Log(("vgdrvDarwinOpen: g_DevExt=%p pSession=%p rc=%d pid=%d\n", &g_DevExt, pSession, rc, proc_pid(pProcess)));
466 return vgdrvDarwinErr2DarwinErr(rc);
467}
468
469
470/**
471 * Close device.
472 */
473static int vgdrvDarwinClose(dev_t Dev, int fFlags, int fDevType, struct proc *pProcess)
474{
475 RT_NOREF(Dev, fFlags, fDevType, pProcess);
476 Log(("vgdrvDarwinClose: pid=%d\n", (int)RTProcSelf()));
477 Assert(proc_pid(pProcess) == (int)RTProcSelf());
478
479 /*
480 * Hand the session closing to org_virtualbox_VBoxGuestClient.
481 */
482 org_virtualbox_VBoxGuestClient::sessionClose(RTProcSelf());
483 return 0;
484}
485
486
487/**
488 * Device I/O Control entry point.
489 *
490 * @returns Darwin for slow IOCtls and VBox status code for the fast ones.
491 * @param Dev The device number (major+minor).
492 * @param iCmd The IOCtl command.
493 * @param pData Pointer to the request data.
494 * @param fFlags Flag saying we're a character device (like we didn't know already).
495 * @param pProcess The process issuing this request.
496 */
497static int vgdrvDarwinIOCtl(dev_t Dev, u_long iCmd, caddr_t pData, int fFlags, struct proc *pProcess)
498{
499 RT_NOREF(Dev, fFlags);
500 const bool fUnrestricted = minor(Dev) == 0;
501 const RTPROCESS Process = proc_pid(pProcess);
502 const unsigned iHash = SESSION_HASH(Process);
503 PVBOXGUESTSESSION pSession;
504
505 /*
506 * Find the session.
507 */
508 RTSpinlockAcquire(g_Spinlock);
509 pSession = g_apSessionHashTab[iHash];
510 while (pSession && (pSession->Process != Process || pSession->fUserSession == fUnrestricted || !pSession->fOpened))
511 pSession = pSession->pNextHash;
512
513 //if (RT_LIKELY(pSession))
514 // supdrvSessionRetain(pSession);
515
516 RTSpinlockRelease(g_Spinlock);
517 if (!pSession)
518 {
519 Log(("VBoxDrvDarwinIOCtl: WHAT?!? pSession == NULL! This must be a mistake... pid=%d iCmd=%#lx\n",
520 (int)Process, iCmd));
521 return EINVAL;
522 }
523
524 /*
525 * Deal with the high-speed IOCtl.
526 */
527 int rc;
528 if (VBGL_IOCTL_IS_FAST(iCmd))
529 rc = VGDrvCommonIoCtlFast(iCmd, &g_DevExt, pSession);
530 else
531 rc = vgdrvDarwinIOCtlSlow(pSession, iCmd, pData, pProcess);
532
533 //supdrvSessionRelease(pSession);
534 return rc;
535}
536
537
538/**
539 * Worker for VBoxDrvDarwinIOCtl that takes the slow IOCtl functions.
540 *
541 * @returns Darwin errno.
542 *
543 * @param pSession The session.
544 * @param iCmd The IOCtl command.
545 * @param pData Pointer to the request data.
546 * @param pProcess The calling process.
547 */
548static int vgdrvDarwinIOCtlSlow(PVBOXGUESTSESSION pSession, u_long iCmd, caddr_t pData, struct proc *pProcess)
549{
550 RT_NOREF(pProcess);
551 LogFlow(("vgdrvDarwinIOCtlSlow: pSession=%p iCmd=%p pData=%p pProcess=%p\n", pSession, iCmd, pData, pProcess));
552
553
554 /*
555 * Buffered or unbuffered?
556 */
557 PVBGLREQHDR pHdr;
558 user_addr_t pUser = 0;
559 void *pvPageBuf = NULL;
560 uint32_t cbReq = IOCPARM_LEN(iCmd);
561 if ((IOC_DIRMASK & iCmd) == IOC_INOUT)
562 {
563 pHdr = (PVBGLREQHDR)pData;
564 if (RT_UNLIKELY(cbReq < sizeof(*pHdr)))
565 {
566 LogRel(("vgdrvDarwinIOCtlSlow: cbReq=%#x < %#x; iCmd=%#lx\n", cbReq, (int)sizeof(*pHdr), iCmd));
567 return EINVAL;
568 }
569 if (RT_UNLIKELY(pHdr->uVersion != VBGLREQHDR_VERSION))
570 {
571 LogRel(("vgdrvDarwinIOCtlSlow: bad uVersion=%#x; iCmd=%#lx\n", pHdr->uVersion, iCmd));
572 return EINVAL;
573 }
574 if (RT_UNLIKELY( RT_MAX(pHdr->cbIn, pHdr->cbOut) != cbReq
575 || pHdr->cbIn < sizeof(*pHdr)
576 || (pHdr->cbOut < sizeof(*pHdr) && pHdr->cbOut != 0)))
577 {
578 LogRel(("vgdrvDarwinIOCtlSlow: max(%#x,%#x) != %#x; iCmd=%#lx\n", pHdr->cbIn, pHdr->cbOut, cbReq, iCmd));
579 return EINVAL;
580 }
581 }
582 else if ((IOC_DIRMASK & iCmd) == IOC_VOID && !cbReq)
583 {
584 /*
585 * Get the header and figure out how much we're gonna have to read.
586 */
587 VBGLREQHDR Hdr;
588 pUser = (user_addr_t)*(void **)pData;
589 int rc = copyin(pUser, &Hdr, sizeof(Hdr));
590 if (RT_UNLIKELY(rc))
591 {
592 LogRel(("vgdrvDarwinIOCtlSlow: copyin(%llx,Hdr,) -> %#x; iCmd=%#lx\n", (unsigned long long)pUser, rc, iCmd));
593 return rc;
594 }
595 if (RT_UNLIKELY(Hdr.uVersion != VBGLREQHDR_VERSION))
596 {
597 LogRel(("vgdrvDarwinIOCtlSlow: bad uVersion=%#x; iCmd=%#lx\n", Hdr.uVersion, iCmd));
598 return EINVAL;
599 }
600 cbReq = RT_MAX(Hdr.cbIn, Hdr.cbOut);
601 if (RT_UNLIKELY( Hdr.cbIn < sizeof(Hdr)
602 || (Hdr.cbOut < sizeof(Hdr) && Hdr.cbOut != 0)
603 || cbReq > _1M*16))
604 {
605 LogRel(("vgdrvDarwinIOCtlSlow: max(%#x,%#x); iCmd=%#lx\n", Hdr.cbIn, Hdr.cbOut, iCmd));
606 return EINVAL;
607 }
608
609 /*
610 * Allocate buffer and copy in the data.
611 */
612 pHdr = (PVBGLREQHDR)RTMemTmpAlloc(cbReq);
613 if (!pHdr)
614 pvPageBuf = pHdr = (PVBGLREQHDR)IOMallocAligned(RT_ALIGN_Z(cbReq, PAGE_SIZE), 8);
615 if (RT_UNLIKELY(!pHdr))
616 {
617 LogRel(("vgdrvDarwinIOCtlSlow: failed to allocate buffer of %d bytes; iCmd=%#lx\n", cbReq, iCmd));
618 return ENOMEM;
619 }
620 rc = copyin(pUser, pHdr, Hdr.cbIn);
621 if (RT_UNLIKELY(rc))
622 {
623 LogRel(("vgdrvDarwinIOCtlSlow: copyin(%llx,%p,%#x) -> %#x; iCmd=%#lx\n",
624 (unsigned long long)pUser, pHdr, Hdr.cbIn, rc, iCmd));
625 if (pvPageBuf)
626 IOFreeAligned(pvPageBuf, RT_ALIGN_Z(cbReq, PAGE_SIZE));
627 else
628 RTMemTmpFree(pHdr);
629 return rc;
630 }
631 if (Hdr.cbIn < cbReq)
632 RT_BZERO((uint8_t *)pHdr + Hdr.cbIn, cbReq - Hdr.cbIn);
633 }
634 else
635 {
636 Log(("vgdrvDarwinIOCtlSlow: huh? cbReq=%#x iCmd=%#lx\n", cbReq, iCmd));
637 return EINVAL;
638 }
639
640 /*
641 * Process the IOCtl.
642 */
643 int rc = VGDrvCommonIoCtl(iCmd, &g_DevExt, pSession, pHdr, cbReq);
644 if (RT_LIKELY(!rc))
645 {
646 /*
647 * If not buffered, copy back the buffer before returning.
648 */
649 if (pUser)
650 {
651 uint32_t cbOut = pHdr->cbOut;
652 if (cbOut > cbReq)
653 {
654 LogRel(("vgdrvDarwinIOCtlSlow: too much output! %#x > %#x; uCmd=%#lx!\n", cbOut, cbReq, iCmd));
655 cbOut = cbReq;
656 }
657 rc = copyout(pHdr, pUser, cbOut);
658 if (RT_UNLIKELY(rc))
659 LogRel(("vgdrvDarwinIOCtlSlow: copyout(%p,%llx,%#x) -> %d; uCmd=%#lx!\n",
660 pHdr, (unsigned long long)pUser, cbOut, rc, iCmd));
661
662 /* cleanup */
663 if (pvPageBuf)
664 IOFreeAligned(pvPageBuf, RT_ALIGN_Z(cbReq, PAGE_SIZE));
665 else
666 RTMemTmpFree(pHdr);
667 }
668 }
669 else
670 {
671 /*
672 * The request failed, just clean up.
673 */
674 if (pUser)
675 {
676 if (pvPageBuf)
677 IOFreeAligned(pvPageBuf, RT_ALIGN_Z(cbReq, PAGE_SIZE));
678 else
679 RTMemTmpFree(pHdr);
680 }
681
682 Log(("vgdrvDarwinIOCtlSlow: pid=%d iCmd=%lx pData=%p failed, rc=%d\n", proc_pid(pProcess), iCmd, (void *)pData, rc));
683 rc = EINVAL;
684 }
685
686 Log2(("vgdrvDarwinIOCtlSlow: returns %d\n", rc));
687 return rc;
688}
689
690
691/**
692 * @note This code is duplicated on other platforms with variations, so please
693 * keep them all up to date when making changes!
694 */
695int VBOXCALL VBoxGuestIDC(void *pvSession, uintptr_t uReq, PVBGLREQHDR pReqHdr, size_t cbReq)
696{
697 /*
698 * Simple request validation (common code does the rest).
699 */
700 int rc;
701 if ( RT_VALID_PTR(pReqHdr)
702 && cbReq >= sizeof(*pReqHdr))
703 {
704 /*
705 * All requests except the connect one requires a valid session.
706 */
707 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pvSession;
708 if (pSession)
709 {
710 if ( RT_VALID_PTR(pSession)
711 && pSession->pDevExt == &g_DevExt)
712 rc = VGDrvCommonIoCtl(uReq, &g_DevExt, pSession, pReqHdr, cbReq);
713 else
714 rc = VERR_INVALID_HANDLE;
715 }
716 else if (uReq == VBGL_IOCTL_IDC_CONNECT)
717 {
718 rc = VGDrvCommonCreateKernelSession(&g_DevExt, &pSession);
719 if (RT_SUCCESS(rc))
720 {
721 rc = VGDrvCommonIoCtl(uReq, &g_DevExt, pSession, pReqHdr, cbReq);
722 if (RT_FAILURE(rc))
723 VGDrvCommonCloseSession(&g_DevExt, pSession);
724 }
725 }
726 else
727 rc = VERR_INVALID_HANDLE;
728 }
729 else
730 rc = VERR_INVALID_POINTER;
731 return rc;
732}
733
734
735void VGDrvNativeISRMousePollEvent(PVBOXGUESTDEVEXT pDevExt)
736{
737 NOREF(pDevExt);
738}
739
740
741bool VGDrvNativeProcessOption(PVBOXGUESTDEVEXT pDevExt, const char *pszName, const char *pszValue)
742{
743 RT_NOREF(pDevExt); RT_NOREF(pszName); RT_NOREF(pszValue);
744 return false;
745}
746
747
748/**
749 * Callback for blah blah blah.
750 *
751 * @todo move to IPRT.
752 */
753static IOReturn vgdrvDarwinSleepHandler(void *pvTarget, void *pvRefCon, UInt32 uMessageType,
754 IOService *pProvider, void *pvMsgArg, vm_size_t cbMsgArg)
755{
756 RT_NOREF(pvTarget, pProvider, pvMsgArg, cbMsgArg);
757 LogFlow(("VBoxGuest: Got sleep/wake notice. Message type was %x\n", uMessageType));
758
759 if (uMessageType == kIOMessageSystemWillSleep)
760 RTPowerSignalEvent(RTPOWEREVENT_SUSPEND);
761 else if (uMessageType == kIOMessageSystemHasPoweredOn)
762 RTPowerSignalEvent(RTPOWEREVENT_RESUME);
763
764 acknowledgeSleepWakeNotification(pvRefCon);
765
766 return 0;
767}
768
769
770/**
771 * Converts an IPRT error code to a darwin error code.
772 *
773 * @returns corresponding darwin error code.
774 * @param rc IPRT status code.
775 */
776static int vgdrvDarwinErr2DarwinErr(int rc)
777{
778 switch (rc)
779 {
780 case VINF_SUCCESS: return 0;
781 case VERR_GENERAL_FAILURE: return EACCES;
782 case VERR_INVALID_PARAMETER: return EINVAL;
783 case VERR_INVALID_MAGIC: return EILSEQ;
784 case VERR_INVALID_HANDLE: return ENXIO;
785 case VERR_INVALID_POINTER: return EFAULT;
786 case VERR_LOCK_FAILED: return ENOLCK;
787 case VERR_ALREADY_LOADED: return EEXIST;
788 case VERR_PERMISSION_DENIED: return EPERM;
789 case VERR_VERSION_MISMATCH: return ENOSYS;
790 }
791
792 return EPERM;
793}
794
795
796/*
797 *
798 * org_virtualbox_VBoxGuest
799 *
800 * - IOService diff resync -
801 * - IOService diff resync -
802 * - IOService diff resync -
803 *
804 */
805
806
807/**
808 * Initialize the object.
809 */
810bool org_virtualbox_VBoxGuest::init(OSDictionary *pDictionary)
811{
812 LogFlow(("IOService::init([%p], %p)\n", this, pDictionary));
813 if (IOService::init(pDictionary))
814 {
815 /* init members. */
816 return true;
817 }
818 return false;
819}
820
821
822/**
823 * Free the object.
824 */
825void org_virtualbox_VBoxGuest::free(void)
826{
827 RTLogBackdoorPrintf("IOService::free([%p])\n", this); /* might go sideways if we use LogFlow() here. weird. */
828 IOService::free();
829}
830
831
832/**
833 * Check if it's ok to start this service.
834 * It's always ok by us, so it's up to IOService to decide really.
835 */
836IOService *org_virtualbox_VBoxGuest::probe(IOService *pProvider, SInt32 *pi32Score)
837{
838 LogFlow(("IOService::probe([%p])\n", this));
839 IOService *pRet = IOService::probe(pProvider, pi32Score);
840 LogFlow(("IOService::probe([%p]) returns %p *pi32Score=%d\n", this, pRet, pi32Score ? *pi32Score : -1));
841 return pRet;
842}
843
844
845/**
846 * Start this service.
847 */
848bool org_virtualbox_VBoxGuest::start(IOService *pProvider)
849{
850 LogFlow(("IOService::start([%p])\n", this));
851
852 /*
853 * Low level initialization / device initialization should be performed only once.
854 */
855 if (ASMAtomicCmpXchgBool(&g_fInstantiated, true, false))
856 {
857 /*
858 * Make sure it's a PCI device.
859 */
860 m_pIOPCIDevice = OSDynamicCast(IOPCIDevice, pProvider);
861 if (m_pIOPCIDevice)
862 {
863 /*
864 * Call parent.
865 */
866 if (IOService::start(pProvider))
867 {
868 /*
869 * Is it the VMM device?
870 */
871 if (isVmmDev(m_pIOPCIDevice))
872 {
873 /*
874 * Enable I/O port and memory regions on the device.
875 */
876 m_pIOPCIDevice->setMemoryEnable(true);
877 m_pIOPCIDevice->setIOEnable(true);
878
879 /*
880 * Region #0: I/O ports. Mandatory.
881 */
882 IOMemoryDescriptor *pMem = m_pIOPCIDevice->getDeviceMemoryWithIndex(0);
883 if (pMem)
884 {
885 IOPhysicalAddress IOPortBasePhys = pMem->getPhysicalAddress();
886 if ((IOPortBasePhys >> 16) == 0)
887 {
888 RTIOPORT IOPortBase = (RTIOPORT)IOPortBasePhys;
889 void *pvMMIOBase = NULL;
890 uint32_t cbMMIO = 0;
891
892 /*
893 * Region #1: Shared Memory. Technically optional.
894 */
895 m_pMap = m_pIOPCIDevice->mapDeviceMemoryWithIndex(1);
896 if (m_pMap)
897 {
898 pvMMIOBase = (void *)m_pMap->getVirtualAddress();
899 cbMMIO = m_pMap->getLength();
900 }
901
902 /*
903 * Initialize the device extension.
904 */
905 int rc = VGDrvCommonInitDevExt(&g_DevExt, IOPortBase, pvMMIOBase, cbMMIO,
906 ARCH_BITS == 64 ? VBOXOSTYPE_MacOS_x64 : VBOXOSTYPE_MacOS, 0);
907 if (RT_SUCCESS(rc))
908 {
909 /*
910 * Register the device nodes and enable interrupts.
911 */
912 rc = vgdrvDarwinCharDevInit();
913 if (rc == KMOD_RETURN_SUCCESS)
914 {
915 if (setupVmmDevInterrupts(pProvider))
916 {
917 /*
918 * Read host configuration.
919 */
920 VGDrvCommonProcessOptionsFromHost(&g_DevExt);
921
922 /*
923 * Just register the service and we're done!
924 */
925 registerService();
926
927 LogRel(("VBoxGuest: IOService started\n"));
928 return true;
929 }
930
931 LogRel(("VBoxGuest: Failed to set up interrupts\n"));
932 vgdrvDarwinCharDevRemove();
933 }
934 else
935 LogRel(("VBoxGuest: Failed to initialize character devices (rc=%#x).\n", rc));
936
937 VGDrvCommonDeleteDevExt(&g_DevExt);
938 }
939 else
940 LogRel(("VBoxGuest: Failed to initialize common code (rc=%Rrc).\n", rc));
941
942 if (m_pMap)
943 {
944 m_pMap->release();
945 m_pMap = NULL;
946 }
947 }
948 else
949 LogRel(("VBoxGuest: Bad I/O port address: %#RX64\n", (uint64_t)IOPortBasePhys));
950 }
951 else
952 LogRel(("VBoxGuest: The device missing is the I/O port range (#0).\n"));
953 }
954 else
955 LogRel(("VBoxGuest: Not the VMMDev (%#x:%#x).\n",
956 m_pIOPCIDevice->configRead16(kIOPCIConfigVendorID), m_pIOPCIDevice->configRead16(kIOPCIConfigDeviceID)));
957
958 IOService::stop(pProvider);
959 }
960 }
961 else
962 LogRel(("VBoxGuest: Provider is not an instance of IOPCIDevice.\n"));
963
964 ASMAtomicXchgBool(&g_fInstantiated, false);
965 }
966 return false;
967}
968
969
970/**
971 * Stop this service.
972 */
973void org_virtualbox_VBoxGuest::stop(IOService *pProvider)
974{
975#ifdef LOG_ENABLED
976 RTLogBackdoorPrintf("org_virtualbox_VBoxGuest::stop([%p], %p)\n", this, pProvider); /* Being cautious here, no Log(). */
977#endif
978 AssertReturnVoid(ASMAtomicReadBool(&g_fInstantiated));
979
980 /* Low level termination should be performed only once */
981 if (!disableVmmDevInterrupts())
982 printf("VBoxGuest: unable to unregister interrupt handler\n");
983
984 vgdrvDarwinCharDevRemove();
985 VGDrvCommonDeleteDevExt(&g_DevExt);
986
987 if (m_pMap)
988 {
989 m_pMap->release();
990 m_pMap = NULL;
991 }
992
993 IOService::stop(pProvider);
994
995 ASMAtomicWriteBool(&g_fInstantiated, false);
996
997 printf("VBoxGuest: IOService stopped\n");
998 RTLogBackdoorPrintf("org_virtualbox_VBoxGuest::stop: returning\n"); /* Being cautious here, no Log(). */
999}
1000
1001
1002/**
1003 * Termination request.
1004 *
1005 * @return true if we're ok with shutting down now, false if we're not.
1006 * @param fOptions Flags.
1007 */
1008bool org_virtualbox_VBoxGuest::terminate(IOOptionBits fOptions)
1009{
1010#ifdef LOG_ENABLED
1011 RTLogBackdoorPrintf("org_virtualbox_VBoxGuest::terminate: reference_count=%d g_cSessions=%d (fOptions=%#x)\n",
1012 KMOD_INFO_NAME.reference_count, ASMAtomicUoReadS32(&g_cSessions), fOptions); /* Being cautious here, no Log(). */
1013#endif
1014
1015 bool fRc;
1016 if ( KMOD_INFO_NAME.reference_count != 0
1017 || ASMAtomicUoReadS32(&g_cSessions))
1018 fRc = false;
1019 else
1020 fRc = IOService::terminate(fOptions);
1021
1022#ifdef LOG_ENABLED
1023 RTLogBackdoorPrintf("org_virtualbox_SupDrv::terminate: returns %d\n", fRc); /* Being cautious here, no Log(). */
1024#endif
1025 return fRc;
1026}
1027
1028
1029/**
1030 * Implementes a IOInterruptHandler, called by provider when an interrupt occurs.
1031 */
1032/*static*/ void org_virtualbox_VBoxGuest::vgdrvDarwinIrqHandler(OSObject *pTarget, void *pvRefCon, IOService *pNub, int iSrc)
1033{
1034#ifdef LOG_ENABLED
1035 RTLogBackdoorPrintf("vgdrvDarwinIrqHandler: %p %p %p %d\n", pTarget, pvRefCon, pNub, iSrc);
1036#endif
1037 RT_NOREF(pTarget, pvRefCon, pNub, iSrc);
1038
1039 VGDrvCommonISR(&g_DevExt);
1040 /* There is in fact no way of indicating that this is our interrupt, other
1041 than making the device lower it. So, the return code is ignored. */
1042}
1043
1044
1045/**
1046 * Sets up and enables interrupts on the device.
1047 *
1048 * Interrupts are handled directly, no messing around with workloops. The
1049 * rational here is is that the main job of our interrupt handler is waking up
1050 * other threads currently sitting in HGCM calls, i.e. little more effort than
1051 * waking up the workloop thread.
1052 *
1053 * @returns success indicator. Failures are fully logged.
1054 */
1055bool org_virtualbox_VBoxGuest::setupVmmDevInterrupts(IOService *pProvider)
1056{
1057 AssertReturn(pProvider, false);
1058
1059 if (m_pInterruptProvider != pProvider)
1060 {
1061 pProvider->retain();
1062 if (m_pInterruptProvider)
1063 m_pInterruptProvider->release();
1064 m_pInterruptProvider = pProvider;
1065 }
1066
1067 IOReturn rc = pProvider->registerInterrupt(0 /*intIndex*/, this, vgdrvDarwinIrqHandler, this);
1068 if (rc == kIOReturnSuccess)
1069 {
1070 rc = pProvider->enableInterrupt(0 /*intIndex*/);
1071 if (rc == kIOReturnSuccess)
1072 return true;
1073
1074 LogRel(("VBoxGuest: Failed to enable interrupt: %#x\n", rc));
1075 m_pInterruptProvider->unregisterInterrupt(0 /*intIndex*/);
1076 }
1077 else
1078 LogRel(("VBoxGuest: Failed to register interrupt: %#x\n", rc));
1079 return false;
1080}
1081
1082
1083/**
1084 * Counterpart to setupVmmDevInterrupts().
1085 */
1086bool org_virtualbox_VBoxGuest::disableVmmDevInterrupts(void)
1087{
1088 if (m_pInterruptProvider)
1089 {
1090 IOReturn rc = m_pInterruptProvider->disableInterrupt(0 /*intIndex*/);
1091 AssertMsg(rc == kIOReturnSuccess, ("%#x\n", rc));
1092 rc = m_pInterruptProvider->unregisterInterrupt(0 /*intIndex*/);
1093 AssertMsg(rc == kIOReturnSuccess, ("%#x\n", rc));
1094 RT_NOREF_PV(rc);
1095
1096 m_pInterruptProvider->release();
1097 m_pInterruptProvider = NULL;
1098 }
1099
1100 return true;
1101}
1102
1103
1104/**
1105 * Checks if it's the VMM device.
1106 *
1107 * @returns true if it is, false if it isn't.
1108 * @param pIOPCIDevice The PCI device we think might be the VMM device.
1109 */
1110bool org_virtualbox_VBoxGuest::isVmmDev(IOPCIDevice *pIOPCIDevice)
1111{
1112 if (pIOPCIDevice)
1113 {
1114 uint16_t idVendor = m_pIOPCIDevice->configRead16(kIOPCIConfigVendorID);
1115 if (idVendor == VMMDEV_VENDORID)
1116 {
1117 uint16_t idDevice = m_pIOPCIDevice->configRead16(kIOPCIConfigDeviceID);
1118 if (idDevice == VMMDEV_DEVICEID)
1119 return true;
1120 }
1121 }
1122 return false;
1123}
1124
1125
1126
1127/*
1128 *
1129 * org_virtualbox_VBoxGuestClient
1130 *
1131 */
1132
1133
1134/**
1135 * Initializer called when the client opens the service.
1136 */
1137bool org_virtualbox_VBoxGuestClient::initWithTask(task_t OwningTask, void *pvSecurityId, UInt32 u32Type)
1138{
1139 LogFlow(("org_virtualbox_VBoxGuestClient::initWithTask([%p], %#x, %p, %#x) (cur pid=%d proc=%p)\n",
1140 this, OwningTask, pvSecurityId, u32Type, RTProcSelf(), RTR0ProcHandleSelf()));
1141 AssertMsg((RTR0PROCESS)OwningTask == RTR0ProcHandleSelf(), ("%p %p\n", OwningTask, RTR0ProcHandleSelf()));
1142
1143 if (!OwningTask)
1144 return false;
1145
1146 if (u32Type != VBOXGUEST_DARWIN_IOSERVICE_COOKIE)
1147 {
1148 VBOX_RETRIEVE_CUR_PROC_NAME(szProcName);
1149 LogRelMax(10, ("org_virtualbox_VBoxGuestClient::initWithTask: Bad cookie %#x (%s)\n", u32Type, szProcName));
1150 return false;
1151 }
1152
1153 if (IOUserClient::initWithTask(OwningTask, pvSecurityId , u32Type))
1154 {
1155 /*
1156 * In theory we have to call task_reference() to make sure that the task is
1157 * valid during the lifetime of this object. The pointer is only used to check
1158 * for the context this object is called in though and never dereferenced
1159 * or passed to anything which might, so we just skip this step.
1160 */
1161 m_Task = OwningTask;
1162 m_pSession = NULL;
1163 m_pProvider = NULL;
1164 return true;
1165 }
1166 return false;
1167}
1168
1169
1170/**
1171 * Start the client service.
1172 */
1173bool org_virtualbox_VBoxGuestClient::start(IOService *pProvider)
1174{
1175 LogFlow(("org_virtualbox_VBoxGuestClient::start([%p], %p) (cur pid=%d proc=%p)\n",
1176 this, pProvider, RTProcSelf(), RTR0ProcHandleSelf() ));
1177 AssertMsgReturn((RTR0PROCESS)m_Task == RTR0ProcHandleSelf(),
1178 ("%p %p\n", m_Task, RTR0ProcHandleSelf()),
1179 false);
1180
1181 if (IOUserClient::start(pProvider))
1182 {
1183 m_pProvider = OSDynamicCast(org_virtualbox_VBoxGuest, pProvider);
1184 if (m_pProvider)
1185 {
1186 Assert(!m_pSession);
1187
1188 /*
1189 * Create a new session.
1190 * Note! We complete the requestor stuff in the open method.
1191 */
1192 int rc = VGDrvCommonCreateUserSession(&g_DevExt, VMMDEV_REQUESTOR_USERMODE, &m_pSession);
1193 if (RT_SUCCESS(rc))
1194 {
1195 m_pSession->fOpened = false;
1196 /* The Uid, Gid and fUnrestricted fields are set on open. */
1197
1198 /*
1199 * Insert it into the hash table, checking that there isn't
1200 * already one for this process first. (One session per proc!)
1201 */
1202 unsigned iHash = SESSION_HASH(m_pSession->Process);
1203 RTSpinlockAcquire(g_Spinlock);
1204
1205 PVBOXGUESTSESSION pCur = g_apSessionHashTab[iHash];
1206 while (pCur && pCur->Process != m_pSession->Process)
1207 pCur = pCur->pNextHash;
1208 if (!pCur)
1209 {
1210 m_pSession->pNextHash = g_apSessionHashTab[iHash];
1211 g_apSessionHashTab[iHash] = m_pSession;
1212 m_pSession->pvVBoxGuestClient = this;
1213 ASMAtomicIncS32(&g_cSessions);
1214 rc = VINF_SUCCESS;
1215 }
1216 else
1217 rc = VERR_ALREADY_LOADED;
1218
1219 RTSpinlockRelease(g_Spinlock);
1220 if (RT_SUCCESS(rc))
1221 {
1222 Log(("org_virtualbox_VBoxGuestClient::start: created session %p for pid %d\n", m_pSession, (int)RTProcSelf()));
1223 return true;
1224 }
1225
1226 LogFlow(("org_virtualbox_VBoxGuestClient::start: already got a session for this process (%p)\n", pCur));
1227 VGDrvCommonCloseSession(&g_DevExt, m_pSession); //supdrvSessionRelease(m_pSession);
1228 }
1229
1230 m_pSession = NULL;
1231 LogFlow(("org_virtualbox_VBoxGuestClient::start: rc=%Rrc from supdrvCreateSession\n", rc));
1232 }
1233 else
1234 LogFlow(("org_virtualbox_VBoxGuestClient::start: %p isn't org_virtualbox_VBoxGuest\n", pProvider));
1235 }
1236 return false;
1237}
1238
1239
1240/**
1241 * Common worker for clientClose and VBoxDrvDarwinClose.
1242 */
1243/* static */ void org_virtualbox_VBoxGuestClient::sessionClose(RTPROCESS Process)
1244{
1245 /*
1246 * Find the session and remove it from the hash table.
1247 *
1248 * Note! Only one session per process. (Both start() and
1249 * vgdrvDarwinOpen makes sure this is so.)
1250 */
1251 const unsigned iHash = SESSION_HASH(Process);
1252 RTSpinlockAcquire(g_Spinlock);
1253 PVBOXGUESTSESSION pSession = g_apSessionHashTab[iHash];
1254 if (pSession)
1255 {
1256 if (pSession->Process == Process)
1257 {
1258 g_apSessionHashTab[iHash] = pSession->pNextHash;
1259 pSession->pNextHash = NULL;
1260 ASMAtomicDecS32(&g_cSessions);
1261 }
1262 else
1263 {
1264 PVBOXGUESTSESSION pPrev = pSession;
1265 pSession = pSession->pNextHash;
1266 while (pSession)
1267 {
1268 if (pSession->Process == Process)
1269 {
1270 pPrev->pNextHash = pSession->pNextHash;
1271 pSession->pNextHash = NULL;
1272 ASMAtomicDecS32(&g_cSessions);
1273 break;
1274 }
1275
1276 /* next */
1277 pPrev = pSession;
1278 pSession = pSession->pNextHash;
1279 }
1280 }
1281 }
1282 RTSpinlockRelease(g_Spinlock);
1283 if (!pSession)
1284 {
1285 Log(("VBoxGuestClient::sessionClose: pSession == NULL, pid=%d; freed already?\n", (int)Process));
1286 return;
1287 }
1288
1289 /*
1290 * Remove it from the client object.
1291 */
1292 org_virtualbox_VBoxGuestClient *pThis = (org_virtualbox_VBoxGuestClient *)pSession->pvVBoxGuestClient;
1293 pSession->pvVBoxGuestClient = NULL;
1294 if (pThis)
1295 {
1296 Assert(pThis->m_pSession == pSession);
1297 pThis->m_pSession = NULL;
1298 }
1299
1300 /*
1301 * Close the session.
1302 */
1303 VGDrvCommonCloseSession(&g_DevExt, pSession); // supdrvSessionRelease(m_pSession);
1304}
1305
1306
1307/**
1308 * Client exits normally.
1309 */
1310IOReturn org_virtualbox_VBoxGuestClient::clientClose(void)
1311{
1312 LogFlow(("org_virtualbox_VBoxGuestClient::clientClose([%p]) (cur pid=%d proc=%p)\n", this, RTProcSelf(), RTR0ProcHandleSelf()));
1313 AssertMsg((RTR0PROCESS)m_Task == RTR0ProcHandleSelf(), ("%p %p\n", m_Task, RTR0ProcHandleSelf()));
1314
1315 /*
1316 * Clean up the session if it's still around.
1317 *
1318 * We cannot rely 100% on close, and in the case of a dead client
1319 * we'll end up hanging inside vm_map_remove() if we postpone it.
1320 */
1321 if (m_pSession)
1322 {
1323 sessionClose(RTProcSelf());
1324 Assert(!m_pSession);
1325 }
1326
1327 m_pProvider = NULL;
1328 terminate();
1329
1330 return kIOReturnSuccess;
1331}
1332
1333
1334/**
1335 * The client exits abnormally / forgets to do cleanups. (logging)
1336 */
1337IOReturn org_virtualbox_VBoxGuestClient::clientDied(void)
1338{
1339 LogFlow(("IOService::clientDied([%p]) m_Task=%p R0Process=%p Process=%d\n", this, m_Task, RTR0ProcHandleSelf(), RTProcSelf()));
1340
1341 /* IOUserClient::clientDied() calls clientClose, so we'll just do the work there. */
1342 return IOUserClient::clientDied();
1343}
1344
1345
1346/**
1347 * Terminate the service (initiate the destruction). (logging)
1348 */
1349bool org_virtualbox_VBoxGuestClient::terminate(IOOptionBits fOptions)
1350{
1351 LogFlow(("IOService::terminate([%p], %#x)\n", this, fOptions));
1352 return IOUserClient::terminate(fOptions);
1353}
1354
1355
1356/**
1357 * The final stage of the client service destruction. (logging)
1358 */
1359bool org_virtualbox_VBoxGuestClient::finalize(IOOptionBits fOptions)
1360{
1361 LogFlow(("IOService::finalize([%p], %#x)\n", this, fOptions));
1362 return IOUserClient::finalize(fOptions);
1363}
1364
1365
1366/**
1367 * Stop the client service. (logging)
1368 */
1369void org_virtualbox_VBoxGuestClient::stop(IOService *pProvider)
1370{
1371 LogFlow(("IOService::stop([%p])\n", this));
1372 IOUserClient::stop(pProvider);
1373}
1374
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