VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuest/VBoxGuest-freebsd.c@ 68704

Last change on this file since 68704 was 68581, checked in by vboxsync, 7 years ago

merging vbglioc r117759: FreeBSD adjustments and fixes (GAs).

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 23.5 KB
Line 
1/* $Id: VBoxGuest-freebsd.c 68581 2017-08-31 12:11:13Z vboxsync $ */
2/** @file
3 * VirtualBox Guest Additions Driver for FreeBSD.
4 */
5
6/*
7 * Copyright (C) 2007-2016 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
18/** @todo r=bird: This must merge with SUPDrv-freebsd.c before long. The two
19 * source files should only differ on prefixes and the extra bits wrt to the
20 * pci device. I.e. it should be diffable so that fixes to one can easily be
21 * applied to the other. */
22
23
24/*********************************************************************************************************************************
25* Header Files *
26*********************************************************************************************************************************/
27#include <sys/param.h>
28#undef PVM
29#include <sys/types.h>
30#include <sys/module.h>
31#include <sys/systm.h>
32#include <sys/errno.h>
33#include <sys/kernel.h>
34#include <sys/fcntl.h>
35#include <sys/conf.h>
36#include <sys/uio.h>
37#include <sys/bus.h>
38#include <sys/poll.h>
39#include <sys/selinfo.h>
40#include <sys/queue.h>
41#include <sys/lock.h>
42#include <sys/lockmgr.h>
43#include <sys/malloc.h>
44#include <sys/file.h>
45#include <sys/rman.h>
46#include <machine/bus.h>
47#include <machine/resource.h>
48#include <dev/pci/pcivar.h>
49#include <dev/pci/pcireg.h>
50
51#include "VBoxGuestInternal.h"
52#include <VBox/version.h>
53#include <VBox/log.h>
54#include <iprt/assert.h>
55#include <iprt/initterm.h>
56#include <iprt/process.h>
57#include <iprt/string.h>
58#include <iprt/mem.h>
59#include <iprt/asm.h>
60
61
62/*********************************************************************************************************************************
63* Defined Constants And Macros *
64*********************************************************************************************************************************/
65/** The module name. */
66#define DEVICE_NAME "vboxguest"
67
68
69/*********************************************************************************************************************************
70* Structures and Typedefs *
71*********************************************************************************************************************************/
72struct VBoxGuestDeviceState
73{
74 /** Resource ID of the I/O port */
75 int iIOPortResId;
76 /** Pointer to the I/O port resource. */
77 struct resource *pIOPortRes;
78 /** Start address of the IO Port. */
79 uint16_t uIOPortBase;
80 /** Resource ID of the MMIO area */
81 int iVMMDevMemResId;
82 /** Pointer to the MMIO resource. */
83 struct resource *pVMMDevMemRes;
84 /** Handle of the MMIO resource. */
85 bus_space_handle_t VMMDevMemHandle;
86 /** Size of the memory area. */
87 bus_size_t VMMDevMemSize;
88 /** Mapping of the register space */
89 void *pMMIOBase;
90 /** IRQ number */
91 int iIrqResId;
92 /** IRQ resource handle. */
93 struct resource *pIrqRes;
94 /** Pointer to the IRQ handler. */
95 void *pfnIrqHandler;
96 /** VMMDev version */
97 uint32_t u32Version;
98};
99
100
101/*********************************************************************************************************************************
102* Internal Functions *
103*********************************************************************************************************************************/
104/*
105 * Character device file handlers.
106 */
107static d_fdopen_t vgdrvFreeBSDOpen;
108static d_close_t vgdrvFreeBSDClose;
109static d_ioctl_t vgdrvFreeBSDIOCtl;
110static int vgdrvFreeBSDIOCtlSlow(PVBOXGUESTSESSION pSession, u_long ulCmd, caddr_t pvData, struct thread *pTd);
111static d_write_t vgdrvFreeBSDWrite;
112static d_read_t vgdrvFreeBSDRead;
113static d_poll_t vgdrvFreeBSDPoll;
114
115/*
116 * IRQ related functions.
117 */
118static void vgdrvFreeBSDRemoveIRQ(device_t pDevice, void *pvState);
119static int vgdrvFreeBSDAddIRQ(device_t pDevice, void *pvState);
120static int vgdrvFreeBSDISR(void *pvState);
121
122
123/*********************************************************************************************************************************
124* Global Variables *
125*********************************************************************************************************************************/
126static MALLOC_DEFINE(M_VBOXGUEST, "vboxguest", "VirtualBox Guest Device Driver");
127
128#ifndef D_NEEDMINOR
129# define D_NEEDMINOR 0
130#endif
131
132/*
133 * The /dev/vboxguest character device entry points.
134 */
135static struct cdevsw g_vgdrvFreeBSDChrDevSW =
136{
137 .d_version = D_VERSION,
138 .d_flags = D_TRACKCLOSE | D_NEEDMINOR,
139 .d_fdopen = vgdrvFreeBSDOpen,
140 .d_close = vgdrvFreeBSDClose,
141 .d_ioctl = vgdrvFreeBSDIOCtl,
142 .d_read = vgdrvFreeBSDRead,
143 .d_write = vgdrvFreeBSDWrite,
144 .d_poll = vgdrvFreeBSDPoll,
145 .d_name = "vboxguest"
146};
147
148/** Device extention & session data association structure. */
149static VBOXGUESTDEVEXT g_DevExt;
150
151/** List of cloned device. Managed by the kernel. */
152static struct clonedevs *g_pvgdrvFreeBSDClones;
153/** The dev_clone event handler tag. */
154static eventhandler_tag g_vgdrvFreeBSDEHTag;
155/** Reference counter */
156static volatile uint32_t cUsers;
157/** selinfo structure used for polling. */
158static struct selinfo g_SelInfo;
159
160/**
161 * DEVFS event handler.
162 */
163static void vgdrvFreeBSDClone(void *pvArg, struct ucred *pCred, char *pszName, int cchName, struct cdev **ppDev)
164{
165 int iUnit;
166 int rc;
167
168 Log(("vgdrvFreeBSDClone: pszName=%s ppDev=%p\n", pszName, ppDev));
169
170 /*
171 * One device node per user, si_drv1 points to the session.
172 * /dev/vboxguest<N> where N = {0...255}.
173 */
174 if (!ppDev)
175 return;
176 if (strcmp(pszName, "vboxguest") == 0)
177 iUnit = -1;
178 else if (dev_stdclone(pszName, NULL, "vboxguest", &iUnit) != 1)
179 return;
180 if (iUnit >= 256)
181 {
182 Log(("vgdrvFreeBSDClone: iUnit=%d >= 256 - rejected\n", iUnit));
183 return;
184 }
185
186 Log(("vgdrvFreeBSDClone: pszName=%s iUnit=%d\n", pszName, iUnit));
187
188 rc = clone_create(&g_pvgdrvFreeBSDClones, &g_vgdrvFreeBSDChrDevSW, &iUnit, ppDev, 0);
189 Log(("vgdrvFreeBSDClone: clone_create -> %d; iUnit=%d\n", rc, iUnit));
190 if (rc)
191 {
192 *ppDev = make_dev(&g_vgdrvFreeBSDChrDevSW,
193 iUnit,
194 UID_ROOT,
195 GID_WHEEL,
196 0664,
197 "vboxguest%d", iUnit);
198 if (*ppDev)
199 {
200 dev_ref(*ppDev);
201 (*ppDev)->si_flags |= SI_CHEAPCLONE;
202 Log(("vgdrvFreeBSDClone: Created *ppDev=%p iUnit=%d si_drv1=%p si_drv2=%p\n",
203 *ppDev, iUnit, (*ppDev)->si_drv1, (*ppDev)->si_drv2));
204 (*ppDev)->si_drv1 = (*ppDev)->si_drv2 = NULL;
205 }
206 else
207 Log(("vgdrvFreeBSDClone: make_dev iUnit=%d failed\n", iUnit));
208 }
209 else
210 Log(("vgdrvFreeBSDClone: Existing *ppDev=%p iUnit=%d si_drv1=%p si_drv2=%p\n",
211 *ppDev, iUnit, (*ppDev)->si_drv1, (*ppDev)->si_drv2));
212}
213
214/**
215 * File open handler
216 *
217 */
218#if __FreeBSD_version >= 700000
219static int vgdrvFreeBSDOpen(struct cdev *pDev, int fOpen, struct thread *pTd, struct file *pFd)
220#else
221static int vgdrvFreeBSDOpen(struct cdev *pDev, int fOpen, struct thread *pTd)
222#endif
223{
224 int rc;
225 PVBOXGUESTSESSION pSession;
226
227 LogFlow(("vgdrvFreeBSDOpen:\n"));
228
229 /*
230 * Try grab it (we don't grab the giant, remember).
231 */
232 if (!ASMAtomicCmpXchgPtr(&pDev->si_drv1, (void *)0x42, NULL))
233 return EBUSY;
234
235 /*
236 * Create a new session.
237 */
238 rc = VGDrvCommonCreateUserSession(&g_DevExt, &pSession);
239 if (RT_SUCCESS(rc))
240 {
241 if (ASMAtomicCmpXchgPtr(&pDev->si_drv1, pSession, (void *)0x42))
242 {
243 Log(("vgdrvFreeBSDOpen: success - g_DevExt=%p pSession=%p rc=%d pid=%d\n", &g_DevExt, pSession, rc, (int)RTProcSelf()));
244 ASMAtomicIncU32(&cUsers);
245 return 0;
246 }
247
248 VGDrvCommonCloseSession(&g_DevExt, pSession);
249 }
250
251 LogRel(("vgdrvFreeBSDOpen: failed. rc=%d\n", rc));
252 return RTErrConvertToErrno(rc);
253}
254
255/**
256 * File close handler
257 *
258 */
259static int vgdrvFreeBSDClose(struct cdev *pDev, int fFile, int DevType, struct thread *pTd)
260{
261 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pDev->si_drv1;
262 Log(("vgdrvFreeBSDClose: fFile=%#x pSession=%p\n", fFile, pSession));
263
264 /*
265 * Close the session if it's still hanging on to the device...
266 */
267 if (VALID_PTR(pSession))
268 {
269 VGDrvCommonCloseSession(&g_DevExt, pSession);
270 if (!ASMAtomicCmpXchgPtr(&pDev->si_drv1, NULL, pSession))
271 Log(("vgdrvFreeBSDClose: si_drv1=%p expected %p!\n", pDev->si_drv1, pSession));
272 ASMAtomicDecU32(&cUsers);
273 /* Don't use destroy_dev here because it may sleep resulting in a hanging user process. */
274 destroy_dev_sched(pDev);
275 }
276 else
277 Log(("vgdrvFreeBSDClose: si_drv1=%p!\n", pSession));
278 return 0;
279}
280
281
282/**
283 * I/O control request.
284 *
285 * @returns depends...
286 * @param pDev The device.
287 * @param ulCmd The command.
288 * @param pvData Pointer to the data.
289 * @param fFile The file descriptor flags.
290 * @param pTd The calling thread.
291 */
292static int vgdrvFreeBSDIOCtl(struct cdev *pDev, u_long ulCmd, caddr_t pvData, int fFile, struct thread *pTd)
293{
294 PVBOXGUESTSESSION pSession;
295 devfs_get_cdevpriv((void **)&pSession);
296
297 /*
298 * Deal with the fast ioctl path first.
299 */
300 if (VBGL_IOCTL_IS_FAST(ulCmd))
301 return VGDrvCommonIoCtlFast(ulCmd, &g_DevExt, pSession);
302
303 return vgdrvFreeBSDIOCtlSlow(pSession, ulCmd, pvData, pTd);
304}
305
306
307/**
308 * Deal with the 'slow' I/O control requests.
309 *
310 * @returns 0 on success, appropriate errno on failure.
311 * @param pSession The session.
312 * @param ulCmd The command.
313 * @param pvData The request data.
314 * @param pTd The calling thread.
315 */
316static int vgdrvFreeBSDIOCtlSlow(PVBOXGUESTSESSION pSession, u_long ulCmd, caddr_t pvData, struct thread *pTd)
317{
318 PVBGLREQHDR pHdr;
319 uint32_t cbReq = IOCPARM_LEN(ulCmd);
320 void *pvUser = NULL;
321
322 /*
323 * Buffered request?
324 */
325 if ((IOC_DIRMASK & ulCmd) == IOC_INOUT)
326 {
327 pHdr = (PVBGLREQHDR)pvData;
328 if (RT_UNLIKELY(cbReq < sizeof(*pHdr)))
329 {
330 LogRel(("vgdrvFreeBSDIOCtlSlow: cbReq=%#x < %#x; ulCmd=%#lx\n", cbReq, (int)sizeof(*pHdr), ulCmd));
331 return EINVAL;
332 }
333 if (RT_UNLIKELY(pHdr->uVersion != VBGLREQHDR_VERSION))
334 {
335 LogRel(("vgdrvFreeBSDIOCtlSlow: bad uVersion=%#x; ulCmd=%#lx\n", pHdr->uVersion, ulCmd));
336 return EINVAL;
337 }
338 if (RT_UNLIKELY( RT_MAX(pHdr->cbIn, pHdr->cbOut) != cbReq
339 || pHdr->cbIn < sizeof(*pHdr)
340 || (pHdr->cbOut < sizeof(*pHdr) && pHdr->cbOut != 0)))
341 {
342 LogRel(("vgdrvFreeBSDIOCtlSlow: max(%#x,%#x) != %#x; ulCmd=%#lx\n", pHdr->cbIn, pHdr->cbOut, cbReq, ulCmd));
343 return EINVAL;
344 }
345 }
346 /*
347 * Big unbuffered request?
348 */
349 else if ((IOC_DIRMASK & ulCmd) == IOC_VOID && !cbReq)
350 {
351 /*
352 * Read the header, validate it and figure out how much that needs to be buffered.
353 */
354 VBGLREQHDR Hdr;
355 pvUser = *(void **)pvData;
356 int rc = copyin(pvUser, &Hdr, sizeof(Hdr));
357 if (RT_UNLIKELY(rc))
358 {
359 LogRel(("vgdrvFreeBSDIOCtlSlow: copyin(%p,Hdr,) -> %#x; ulCmd=%#lx\n", pvUser, rc, ulCmd));
360 return rc;
361 }
362 if (RT_UNLIKELY(Hdr.uVersion != VBGLREQHDR_VERSION))
363 {
364 LogRel(("vgdrvFreeBSDIOCtlSlow: bad uVersion=%#x; ulCmd=%#lx\n", Hdr.uVersion, ulCmd));
365 return EINVAL;
366 }
367 cbReq = RT_MAX(Hdr.cbIn, Hdr.cbOut);
368 if (RT_UNLIKELY( Hdr.cbIn < sizeof(Hdr)
369 || (Hdr.cbOut < sizeof(Hdr) && Hdr.cbOut != 0)
370 || cbReq > _1M*16))
371 {
372 LogRel(("vgdrvFreeBSDIOCtlSlow: max(%#x,%#x); ulCmd=%#lx\n", Hdr.cbIn, Hdr.cbOut, ulCmd));
373 return EINVAL;
374 }
375
376 /*
377 * Allocate buffer and copy in the data.
378 */
379 pHdr = (PVBGLREQHDR)RTMemTmpAlloc(cbReq);
380 if (RT_UNLIKELY(!pHdr))
381 {
382 LogRel(("vgdrvFreeBSDIOCtlSlow: failed to allocate buffer of %d bytes; ulCmd=%#lx\n", cbReq, ulCmd));
383 return ENOMEM;
384 }
385 rc = copyin(pvUser, pHdr, Hdr.cbIn);
386 if (RT_UNLIKELY(rc))
387 {
388 LogRel(("vgdrvFreeBSDIOCtlSlow: copyin(%p,%p,%#x) -> %#x; ulCmd=%#lx\n",
389 pvUser, pHdr, Hdr.cbIn, rc, ulCmd));
390 RTMemTmpFree(pHdr);
391 return rc;
392 }
393 if (Hdr.cbIn < cbReq)
394 RT_BZERO((uint8_t *)pHdr + Hdr.cbIn, cbReq - Hdr.cbIn);
395 }
396 else
397 {
398 Log(("vgdrvFreeBSDIOCtlSlow: huh? cbReq=%#x ulCmd=%#lx\n", cbReq, ulCmd));
399 return EINVAL;
400 }
401
402 /*
403 * Process the IOCtl.
404 */
405 int rc = VGDrvCommonIoCtl(ulCmd, &g_DevExt, pSession, pHdr, cbReq);
406 if (RT_LIKELY(!rc))
407 {
408 /*
409 * If unbuffered, copy back the result before returning.
410 */
411 if (pvUser)
412 {
413 uint32_t cbOut = pHdr->cbOut;
414 if (cbOut > cbReq)
415 {
416 LogRel(("vgdrvFreeBSDIOCtlSlow: too much output! %#x > %#x; uCmd=%#lx!\n", cbOut, cbReq, ulCmd));
417 cbOut = cbReq;
418 }
419 rc = copyout(pHdr, pvUser, cbOut);
420 if (RT_UNLIKELY(rc))
421 LogRel(("vgdrvFreeBSDIOCtlSlow: copyout(%p,%p,%#x) -> %d; uCmd=%#lx!\n", pHdr, pvUser, cbOut, rc, ulCmd));
422
423 Log(("vgdrvFreeBSDIOCtlSlow: returns %d / %d ulCmd=%lx\n", 0, pHdr->rc, ulCmd));
424
425 /* cleanup */
426 RTMemTmpFree(pHdr);
427 }
428 }
429 else
430 {
431 /*
432 * The request failed, just clean up.
433 */
434 if (pvUser)
435 RTMemTmpFree(pHdr);
436
437 Log(("vgdrvFreeBSDIOCtlSlow: ulCmd=%lx pData=%p failed, rc=%d\n", ulCmd, pvData, rc));
438 rc = EINVAL;
439 }
440
441 return rc;
442}
443
444
445/**
446 * @note This code is duplicated on other platforms with variations, so please
447 * keep them all up to date when making changes!
448 */
449int VBOXCALL VBoxGuestIDC(void *pvSession, uintptr_t uReq, PVBGLREQHDR pReqHdr, size_t cbReq)
450{
451 /*
452 * Simple request validation (common code does the rest).
453 */
454 int rc;
455 if ( RT_VALID_PTR(pReqHdr)
456 && cbReq >= sizeof(*pReqHdr))
457 {
458 /*
459 * All requests except the connect one requires a valid session.
460 */
461 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pvSession;
462 if (pSession)
463 {
464 if ( RT_VALID_PTR(pSession)
465 && pSession->pDevExt == &g_DevExt)
466 rc = VGDrvCommonIoCtl(uReq, &g_DevExt, pSession, pReqHdr, cbReq);
467 else
468 rc = VERR_INVALID_HANDLE;
469 }
470 else if (uReq == VBGL_IOCTL_IDC_CONNECT)
471 {
472 rc = VGDrvCommonCreateKernelSession(&g_DevExt, &pSession);
473 if (RT_SUCCESS(rc))
474 {
475 rc = VGDrvCommonIoCtl(uReq, &g_DevExt, pSession, pReqHdr, cbReq);
476 if (RT_FAILURE(rc))
477 VGDrvCommonCloseSession(&g_DevExt, pSession);
478 }
479 }
480 else
481 rc = VERR_INVALID_HANDLE;
482 }
483 else
484 rc = VERR_INVALID_POINTER;
485 return rc;
486}
487
488
489static int vgdrvFreeBSDPoll(struct cdev *pDev, int fEvents, struct thread *td)
490{
491 int fEventsProcessed;
492
493 LogFlow(("vgdrvFreeBSDPoll: fEvents=%d\n", fEvents));
494
495 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pDev->si_drv1;
496 if (RT_UNLIKELY(!VALID_PTR(pSession))) {
497 Log(("vgdrvFreeBSDPoll: no state data for %s\n", devtoname(pDev)));
498 return (fEvents & (POLLHUP|POLLIN|POLLRDNORM|POLLOUT|POLLWRNORM));
499 }
500
501 uint32_t u32CurSeq = ASMAtomicUoReadU32(&g_DevExt.u32MousePosChangedSeq);
502 if (pSession->u32MousePosChangedSeq != u32CurSeq)
503 {
504 fEventsProcessed = fEvents & (POLLIN | POLLRDNORM);
505 pSession->u32MousePosChangedSeq = u32CurSeq;
506 }
507 else
508 {
509 fEventsProcessed = 0;
510
511 selrecord(td, &g_SelInfo);
512 }
513
514 return fEventsProcessed;
515}
516
517static int vgdrvFreeBSDWrite(struct cdev *pDev, struct uio *pUio, int fIo)
518{
519 return 0;
520}
521
522static int vgdrvFreeBSDRead(struct cdev *pDev, struct uio *pUio, int fIo)
523{
524 return 0;
525}
526
527static int vgdrvFreeBSDDetach(device_t pDevice)
528{
529 struct VBoxGuestDeviceState *pState = device_get_softc(pDevice);
530
531 if (cUsers > 0)
532 return EBUSY;
533
534 /*
535 * Reverse what we did in vgdrvFreeBSDAttach.
536 */
537 if (g_vgdrvFreeBSDEHTag != NULL)
538 EVENTHANDLER_DEREGISTER(dev_clone, g_vgdrvFreeBSDEHTag);
539
540 clone_cleanup(&g_pvgdrvFreeBSDClones);
541
542 vgdrvFreeBSDRemoveIRQ(pDevice, pState);
543
544 if (pState->pVMMDevMemRes)
545 bus_release_resource(pDevice, SYS_RES_MEMORY, pState->iVMMDevMemResId, pState->pVMMDevMemRes);
546 if (pState->pIOPortRes)
547 bus_release_resource(pDevice, SYS_RES_IOPORT, pState->iIOPortResId, pState->pIOPortRes);
548
549 VGDrvCommonDeleteDevExt(&g_DevExt);
550
551 RTR0Term();
552
553 return 0;
554}
555
556/**
557 * Interrupt service routine.
558 *
559 * @returns Whether the interrupt was from VMMDev.
560 * @param pvState Opaque pointer to the device state.
561 */
562static int vgdrvFreeBSDISR(void *pvState)
563{
564 LogFlow(("vgdrvFreeBSDISR: pvState=%p\n", pvState));
565
566 bool fOurIRQ = VGDrvCommonISR(&g_DevExt);
567
568 return fOurIRQ ? 0 : 1;
569}
570
571void VGDrvNativeISRMousePollEvent(PVBOXGUESTDEVEXT pDevExt)
572{
573 LogFlow(("VGDrvNativeISRMousePollEvent:\n"));
574
575 /*
576 * Wake up poll waiters.
577 */
578 selwakeup(&g_SelInfo);
579}
580
581/**
582 * Sets IRQ for VMMDev.
583 *
584 * @returns FreeBSD error code.
585 * @param pDevice Pointer to the device info structure.
586 * @param pvState Pointer to the state info structure.
587 */
588static int vgdrvFreeBSDAddIRQ(device_t pDevice, void *pvState)
589{
590 int iResId = 0;
591 int rc = 0;
592 struct VBoxGuestDeviceState *pState = (struct VBoxGuestDeviceState *)pvState;
593
594 pState->pIrqRes = bus_alloc_resource_any(pDevice, SYS_RES_IRQ, &iResId, RF_SHAREABLE | RF_ACTIVE);
595
596#if __FreeBSD_version >= 700000
597 rc = bus_setup_intr(pDevice, pState->pIrqRes, INTR_TYPE_BIO | INTR_MPSAFE, NULL, (driver_intr_t *)vgdrvFreeBSDISR, pState,
598 &pState->pfnIrqHandler);
599#else
600 rc = bus_setup_intr(pDevice, pState->pIrqRes, INTR_TYPE_BIO, (driver_intr_t *)vgdrvFreeBSDISR, pState, &pState->pfnIrqHandler);
601#endif
602
603 if (rc)
604 {
605 pState->pfnIrqHandler = NULL;
606 return VERR_DEV_IO_ERROR;
607 }
608
609 pState->iIrqResId = iResId;
610
611 return VINF_SUCCESS;
612}
613
614/**
615 * Removes IRQ for VMMDev.
616 *
617 * @param pDevice Pointer to the device info structure.
618 * @param pvState Opaque pointer to the state info structure.
619 */
620static void vgdrvFreeBSDRemoveIRQ(device_t pDevice, void *pvState)
621{
622 struct VBoxGuestDeviceState *pState = (struct VBoxGuestDeviceState *)pvState;
623
624 if (pState->pIrqRes)
625 {
626 bus_teardown_intr(pDevice, pState->pIrqRes, pState->pfnIrqHandler);
627 bus_release_resource(pDevice, SYS_RES_IRQ, 0, pState->pIrqRes);
628 }
629}
630
631static int vgdrvFreeBSDAttach(device_t pDevice)
632{
633 int rc;
634 int iResId;
635 struct VBoxGuestDeviceState *pState;
636
637 cUsers = 0;
638
639 /*
640 * Initialize IPRT R0 driver, which internally calls OS-specific r0 init.
641 */
642 rc = RTR0Init(0);
643 if (RT_FAILURE(rc))
644 {
645 LogFunc(("RTR0Init failed.\n"));
646 return ENXIO;
647 }
648
649 pState = device_get_softc(pDevice);
650
651 /*
652 * Allocate I/O port resource.
653 */
654 iResId = PCIR_BAR(0);
655 pState->pIOPortRes = bus_alloc_resource_any(pDevice, SYS_RES_IOPORT, &iResId, RF_ACTIVE);
656 pState->uIOPortBase = rman_get_start(pState->pIOPortRes);
657 pState->iIOPortResId = iResId;
658 if (pState->uIOPortBase)
659 {
660 /*
661 * Map the MMIO region.
662 */
663 iResId = PCIR_BAR(1);
664 pState->pVMMDevMemRes = bus_alloc_resource_any(pDevice, SYS_RES_MEMORY, &iResId, RF_ACTIVE);
665 pState->VMMDevMemHandle = rman_get_bushandle(pState->pVMMDevMemRes);
666 pState->VMMDevMemSize = rman_get_size(pState->pVMMDevMemRes);
667
668 pState->pMMIOBase = rman_get_virtual(pState->pVMMDevMemRes);
669 pState->iVMMDevMemResId = iResId;
670 if (pState->pMMIOBase)
671 {
672 /*
673 * Call the common device extension initializer.
674 */
675 rc = VGDrvCommonInitDevExt(&g_DevExt, pState->uIOPortBase,
676 pState->pMMIOBase, pState->VMMDevMemSize,
677#if ARCH_BITS == 64
678 VBOXOSTYPE_FreeBSD_x64,
679#else
680 VBOXOSTYPE_FreeBSD,
681#endif
682 VMMDEV_EVENT_MOUSE_POSITION_CHANGED);
683 if (RT_SUCCESS(rc))
684 {
685 /*
686 * Add IRQ of VMMDev.
687 */
688 rc = vgdrvFreeBSDAddIRQ(pDevice, pState);
689 if (RT_SUCCESS(rc))
690 {
691 /*
692 * Configure device cloning.
693 */
694 clone_setup(&g_pvgdrvFreeBSDClones);
695 g_vgdrvFreeBSDEHTag = EVENTHANDLER_REGISTER(dev_clone, vgdrvFreeBSDClone, 0, 1000);
696 if (g_vgdrvFreeBSDEHTag)
697 {
698 printf(DEVICE_NAME ": loaded successfully\n");
699 return 0;
700 }
701
702 printf(DEVICE_NAME ": EVENTHANDLER_REGISTER(dev_clone,,,) failed\n");
703 clone_cleanup(&g_pvgdrvFreeBSDClones);
704 vgdrvFreeBSDRemoveIRQ(pDevice, pState);
705 }
706 else
707 printf((DEVICE_NAME ": VGDrvCommonInitDevExt failed.\n"));
708 VGDrvCommonDeleteDevExt(&g_DevExt);
709 }
710 else
711 printf((DEVICE_NAME ": vgdrvFreeBSDAddIRQ failed.\n"));
712 }
713 else
714 printf((DEVICE_NAME ": MMIO region setup failed.\n"));
715 }
716 else
717 printf((DEVICE_NAME ": IOport setup failed.\n"));
718
719 RTR0Term();
720 return ENXIO;
721}
722
723static int vgdrvFreeBSDProbe(device_t pDevice)
724{
725 if ((pci_get_vendor(pDevice) == VMMDEV_VENDORID) && (pci_get_device(pDevice) == VMMDEV_DEVICEID))
726 return 0;
727
728 return ENXIO;
729}
730
731static device_method_t vgdrvFreeBSDMethods[] =
732{
733 /* Device interface. */
734 DEVMETHOD(device_probe, vgdrvFreeBSDProbe),
735 DEVMETHOD(device_attach, vgdrvFreeBSDAttach),
736 DEVMETHOD(device_detach, vgdrvFreeBSDDetach),
737 {0,0}
738};
739
740static driver_t vgdrvFreeBSDDriver =
741{
742 DEVICE_NAME,
743 vgdrvFreeBSDMethods,
744 sizeof(struct VBoxGuestDeviceState),
745};
746
747static devclass_t vgdrvFreeBSDClass;
748
749DRIVER_MODULE(vboxguest, pci, vgdrvFreeBSDDriver, vgdrvFreeBSDClass, 0, 0);
750MODULE_VERSION(vboxguest, 1);
751
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