VirtualBox

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

Last change on this file since 24323 was 22583, checked in by vboxsync, 15 years ago

Additions/FreeBSD: Remove a method which is not available in FreeBSD 8 anymore

  • Property svn:eol-style set to native
  • Property svn:keywords set to Id
File size: 18.8 KB
Line 
1/* $Id: VBoxGuest-freebsd.c 22583 2009-08-30 20:59:07Z vboxsync $ */
2/** @file
3 * VirtualBox Guest Additions Driver for FreeBSD.
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/** @todo r=bird: This must merge with SUPDrv-freebsd.c before long. The two
23 * source files should only differ on prefixes and the extra bits wrt to the
24 * pci device. I.e. it should be diffable so that fixes to one can easily be
25 * applied to the other. */
26
27#include <sys/param.h>
28#include <sys/systm.h>
29#include <sys/conf.h>
30#include <sys/kernel.h>
31#include <sys/module.h>
32#include <sys/bus.h>
33#include <sys/poll.h>
34#include <sys/selinfo.h>
35#include <sys/queue.h>
36#include <sys/lock.h>
37#include <sys/lockmgr.h>
38#include <sys/types.h>
39#include <sys/conf.h>
40#include <sys/malloc.h>
41#include <sys/uio.h>
42#include <sys/file.h>
43#include <sys/rman.h>
44#include <machine/bus.h>
45#include <machine/resource.h>
46#include <dev/pci/pcivar.h>
47#include <dev/pci/pcireg.h>
48
49#ifdef PVM
50# undef PVM
51#endif
52#include "VBoxGuestInternal.h"
53#include <VBox/log.h>
54#include <iprt/assert.h>
55#include <iprt/initterm.h>
56#include <iprt/process.h>
57#include <iprt/mem.h>
58#include <iprt/asm.h>
59
60/** The module name. */
61#define DEVICE_NAME "vboxguest"
62
63struct VBoxGuestDeviceState
64{
65 /** Resource ID of the I/O port */
66 int iIOPortResId;
67 /** Pointer to the I/O port resource. */
68 struct resource *pIOPortRes;
69 /** Start address of the IO Port. */
70 uint16_t uIOPortBase;
71 /** Resource ID of the MMIO area */
72 int iVMMDevMemResId;
73 /** Pointer to the MMIO resource. */
74 struct resource *pVMMDevMemRes;
75 /** Handle of the MMIO resource. */
76 bus_space_handle_t VMMDevMemHandle;
77 /** Size of the memory area. */
78 bus_size_t VMMDevMemSize;
79 /** Mapping of the register space */
80 void *pMMIOBase;
81 /** IRQ number */
82 int iIrqResId;
83 /** IRQ resource handle. */
84 struct resource *pIrqRes;
85 /** Pointer to the IRQ handler. */
86 void *pfnIrqHandler;
87 /** VMMDev version */
88 uint32_t u32Version;
89};
90
91static MALLOC_DEFINE(M_VBOXDEV, "vboxdev_pci", "VirtualBox Guest driver PCI");
92
93/*
94 * Character device file handlers.
95 */
96static d_fdopen_t VBoxGuestFreeBSDOpen;
97static d_close_t VBoxGuestFreeBSDClose;
98static d_ioctl_t VBoxGuestFreeBSDIOCtl;
99static d_write_t VBoxGuestFreeBSDWrite;
100static d_read_t VBoxGuestFreeBSDRead;
101static d_poll_t VBoxGuestFreeBSDPoll;
102
103/*
104 * IRQ related functions.
105 */
106static void VBoxGuestFreeBSDRemoveIRQ(device_t pDevice, void *pvState);
107static int VBoxGuestFreeBSDAddIRQ(device_t pDevice, void *pvState);
108static int VBoxGuestFreeBSDISR(void *pvState);
109
110/*
111 * Available functions for kernel drivers.
112 */
113DECLVBGL(int) VBoxGuestFreeBSDServiceCall(void *pvSession, unsigned uCmd, void *pvData, size_t cbData, size_t *pcbDataReturned);
114DECLVBGL(void *) VBoxGuestFreeBSDServiceOpen(uint32_t *pu32Version);
115DECLVBGL(int) VBoxGuestFreeBSDServiceClose(void *pvSession);
116
117#ifndef D_NEEDMINOR
118# define D_NEEDMINOR 0
119#endif
120
121/*
122 * Device node entry points.
123 */
124static struct cdevsw g_VBoxGuestFreeBSDChrDevSW =
125{
126 .d_version = D_VERSION,
127 .d_flags = D_TRACKCLOSE | D_NEEDMINOR,
128 .d_fdopen = VBoxGuestFreeBSDOpen,
129 .d_close = VBoxGuestFreeBSDClose,
130 .d_ioctl = VBoxGuestFreeBSDIOCtl,
131 .d_read = VBoxGuestFreeBSDRead,
132 .d_write = VBoxGuestFreeBSDWrite,
133 .d_poll = VBoxGuestFreeBSDPoll,
134 .d_name = DEVICE_NAME
135};
136
137/** Device extention & session data association structure. */
138static VBOXGUESTDEVEXT g_DevExt;
139/** List of cloned device. Managed by the kernel. */
140static struct clonedevs *g_pVBoxGuestFreeBSDClones;
141/** The dev_clone event handler tag. */
142static eventhandler_tag g_VBoxGuestFreeBSDEHTag;
143/** Reference counter */
144static volatile uint32_t cUsers;
145/** selinfo structure used for polling. */
146static struct selinfo g_SelInfo;
147
148/**
149 * DEVFS event handler.
150 */
151static void VBoxGuestFreeBSDClone(void *pvArg, struct ucred *pCred, char *pszName, int cchName, struct cdev **ppDev)
152{
153 int iUnit;
154 int rc;
155
156 Log(("VBoxGuestFreeBSDClone: pszName=%s ppDev=%p\n", pszName, ppDev));
157
158 /*
159 * One device node per user, si_drv1 points to the session.
160 * /dev/vboxguest<N> where N = {0...255}.
161 */
162 if (!ppDev)
163 return;
164 if (strcmp(pszName, "vboxguest") == 0)
165 iUnit = -1;
166 else if (dev_stdclone(pszName, NULL, "vboxguest", &iUnit) != 1)
167 return;
168 if (iUnit >= 256)
169 {
170 Log(("VBoxGuestFreeBSDClone: iUnit=%d >= 256 - rejected\n", iUnit));
171 return;
172 }
173
174 Log(("VBoxGuestFreeBSDClone: pszName=%s iUnit=%d\n", pszName, iUnit));
175
176 rc = clone_create(&g_pVBoxGuestFreeBSDClones, &g_VBoxGuestFreeBSDChrDevSW, &iUnit, ppDev, 0);
177 Log(("VBoxGuestFreeBSDClone: clone_create -> %d; iUnit=%d\n", rc, iUnit));
178 if (rc)
179 {
180 *ppDev = make_dev(&g_VBoxGuestFreeBSDChrDevSW,
181 iUnit,
182 UID_ROOT,
183 GID_WHEEL,
184 0644,
185 "vboxguest%d", iUnit);
186 if (*ppDev)
187 {
188 dev_ref(*ppDev);
189 (*ppDev)->si_flags |= SI_CHEAPCLONE;
190 Log(("VBoxGuestFreeBSDClone: Created *ppDev=%p iUnit=%d si_drv1=%p si_drv2=%p\n",
191 *ppDev, iUnit, (*ppDev)->si_drv1, (*ppDev)->si_drv2));
192 (*ppDev)->si_drv1 = (*ppDev)->si_drv2 = NULL;
193 }
194 else
195 Log(("VBoxGuestFreeBSDClone: make_dev iUnit=%d failed\n", iUnit));
196 }
197 else
198 Log(("VBoxGuestFreeBSDClone: Existing *ppDev=%p iUnit=%d si_drv1=%p si_drv2=%p\n",
199 *ppDev, iUnit, (*ppDev)->si_drv1, (*ppDev)->si_drv2));
200}
201
202/**
203 * File open handler
204 *
205 */
206#if __FreeBSD_version >= 700000
207static int VBoxGuestFreeBSDOpen(struct cdev *pDev, int fOpen, struct thread *pTd, struct file *pFd)
208#else
209static int VBoxGuestFreeBSDOpen(struct cdev *pDev, int fOpen, struct thread *pTd)
210#endif
211{
212 int rc;
213 PVBOXGUESTSESSION pSession;
214
215 LogFlow((DEVICE_NAME ":VBoxGuestFreeBSDOpen\n"));
216
217 /*
218 * Try grab it (we don't grab the giant, remember).
219 */
220 if (!ASMAtomicCmpXchgPtr(&pDev->si_drv1, (void *)0x42, NULL))
221 return EBUSY;
222
223 /*
224 * Create a new session.
225 */
226 rc = VBoxGuestCreateUserSession(&g_DevExt, &pSession);
227 if (RT_SUCCESS(rc))
228 {
229 if (ASMAtomicCmpXchgPtr(&pDev->si_drv1, pSession, (void *)0x42))
230 {
231 Log((DEVICE_NAME ":VBoxGuestFreeBSDOpen success: g_DevExt=%p pSession=%p rc=%d pid=%d\n", &g_DevExt, pSession, rc, (int)RTProcSelf()));
232 ASMAtomicIncU32(&cUsers);
233 return 0;
234 }
235
236 VBoxGuestCloseSession(&g_DevExt, pSession);
237 }
238
239 LogRel((DEVICE_NAME ":VBoxGuestFreeBSDOpen: failed. rc=%d\n", rc));
240 return RTErrConvertToErrno(rc);
241}
242
243/**
244 * File close handler
245 *
246 */
247static int VBoxGuestFreeBSDClose(struct cdev *pDev, int fFile, int DevType, struct thread *pTd)
248{
249 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pDev->si_drv1;
250 Log(("VBoxGuestFreeBSDClose: fFile=%#x pSession=%p\n", fFile, pSession));
251
252 /*
253 * Close the session if it's still hanging on to the device...
254 */
255 if (VALID_PTR(pSession))
256 {
257 VBoxGuestCloseSession(&g_DevExt, pSession);
258 if (!ASMAtomicCmpXchgPtr(&pDev->si_drv1, NULL, pSession))
259 Log(("VBoxGuestFreeBSDClose: si_drv1=%p expected %p!\n", pDev->si_drv1, pSession));
260 ASMAtomicDecU32(&cUsers);
261 /* Don't use destroy_dev here because it may sleep resulting in a hanging user process. */
262 destroy_dev_sched(pDev);
263 }
264 else
265 Log(("VBoxGuestFreeBSDClose: si_drv1=%p!\n", pSession));
266 return 0;
267}
268
269/**
270 * IOCTL handler
271 *
272 */
273static int VBoxGuestFreeBSDIOCtl(struct cdev *pDev, u_long ulCmd, caddr_t pvData, int fFile, struct thread *pTd)
274{
275 LogFlow((DEVICE_NAME ":VBoxGuestFreeBSDIOCtl\n"));
276
277 int rc = 0;
278
279 /*
280 * Validate the input.
281 */
282 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pDev->si_drv1;
283 if (RT_UNLIKELY(!VALID_PTR(pSession)))
284 return EINVAL;
285
286 /*
287 * Validate the request wrapper.
288 */
289 if (IOCPARM_LEN(ulCmd) != sizeof(VBGLBIGREQ))
290 {
291 Log((DEVICE_NAME ": VBoxGuestFreeBSDIOCtl: bad request %lu size=%lu expected=%d\n", ulCmd, IOCPARM_LEN(ulCmd), sizeof(VBGLBIGREQ)));
292 return ENOTTY;
293 }
294
295 PVBGLBIGREQ ReqWrap = (PVBGLBIGREQ)pvData;
296 if (ReqWrap->u32Magic != VBGLBIGREQ_MAGIC)
297 {
298 Log((DEVICE_NAME ": VBoxGuestFreeBSDIOCtl: bad magic %#x; pArg=%p Cmd=%lu.\n", ReqWrap->u32Magic, pvData, ulCmd));
299 return EINVAL;
300 }
301 if (RT_UNLIKELY( ReqWrap->cbData == 0
302 || ReqWrap->cbData > _1M*16))
303 {
304 printf(DEVICE_NAME ": VBoxGuestFreeBSDIOCtl: bad size %#x; pArg=%p Cmd=%lu.\n", ReqWrap->cbData, pvData, ulCmd);
305 return EINVAL;
306 }
307
308 /*
309 * Read the request.
310 */
311 void *pvBuf = RTMemTmpAlloc(ReqWrap->cbData);
312 if (RT_UNLIKELY(!pvBuf))
313 {
314 Log((DEVICE_NAME ":VBoxGuestFreeBSDIOCtl: RTMemTmpAlloc failed to alloc %d bytes.\n", ReqWrap->cbData));
315 return ENOMEM;
316 }
317
318 rc = copyin((void *)(uintptr_t)ReqWrap->pvDataR3, pvBuf, ReqWrap->cbData);
319 if (RT_UNLIKELY(rc))
320 {
321 RTMemTmpFree(pvBuf);
322 Log((DEVICE_NAME ":VBoxGuestFreeBSDIOCtl: copyin failed; pvBuf=%p pArg=%p Cmd=%lu. rc=%d\n", pvBuf, pvData, ulCmd, rc));
323 return EFAULT;
324 }
325 if (RT_UNLIKELY( ReqWrap->cbData != 0
326 && !VALID_PTR(pvBuf)))
327 {
328 RTMemTmpFree(pvBuf);
329 Log((DEVICE_NAME ":VBoxGuestFreeBSDIOCtl: pvBuf invalid pointer %p\n", pvBuf));
330 return EINVAL;
331 }
332 Log((DEVICE_NAME ":VBoxGuestFreeBSDIOCtl: pSession=%p pid=%d.\n", pSession, (int)RTProcSelf()));
333
334 /*
335 * Process the IOCtl.
336 */
337 size_t cbDataReturned;
338 rc = VBoxGuestCommonIOCtl(ulCmd, &g_DevExt, pSession, pvBuf, ReqWrap->cbData, &cbDataReturned);
339 if (RT_SUCCESS(rc))
340 {
341 rc = 0;
342 if (RT_UNLIKELY(cbDataReturned > ReqWrap->cbData))
343 {
344 Log((DEVICE_NAME ":VBoxGuestFreeBSDIOCtl: too much output data %d expected %d\n", cbDataReturned, ReqWrap->cbData));
345 cbDataReturned = ReqWrap->cbData;
346 }
347 if (cbDataReturned > 0)
348 {
349 rc = copyout(pvBuf, (void *)(uintptr_t)ReqWrap->pvDataR3, cbDataReturned);
350 if (RT_UNLIKELY(rc))
351 {
352 Log((DEVICE_NAME ":VBoxGuestFreeBSDIOCtl: copyout failed; pvBuf=%p pArg=%p Cmd=%lu. rc=%d\n", pvBuf, pvData, ulCmd, rc));
353 rc = EFAULT;
354 }
355 }
356 }
357 else
358 {
359 Log((DEVICE_NAME ":VBoxGuestFreeBSDIOCtl: VBoxGuestCommonIOCtl failed. rc=%d\n", rc));
360 rc = EFAULT;
361 }
362 RTMemTmpFree(pvBuf);
363 return rc;
364}
365
366static int VBoxGuestFreeBSDPoll (struct cdev *pDev, int fEvents, struct thread *td)
367{
368 int fEventsProcessed;
369
370 LogFlow((DEVICE_NAME "::Poll: fEvents=%d\n", fEvents));
371
372 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pDev->si_drv1;
373 if (RT_UNLIKELY(!VALID_PTR(pSession))) {
374 Log((DEVICE_NAME "::Poll: no state data for %s\n", devtoname(pDev)));
375 return (fEvents & (POLLHUP|POLLIN|POLLRDNORM|POLLOUT|POLLWRNORM));
376 }
377
378 uint32_t u32CurSeq = ASMAtomicUoReadU32(&g_DevExt.u32MousePosChangedSeq);
379 if (pSession->u32MousePosChangedSeq != u32CurSeq)
380 {
381 fEventsProcessed = fEvents & (POLLIN | POLLRDNORM);
382 pSession->u32MousePosChangedSeq = u32CurSeq;
383 }
384 else
385 {
386 fEventsProcessed = 0;
387
388 selrecord(td, &g_SelInfo);
389 }
390
391 return fEventsProcessed;
392}
393
394static int VBoxGuestFreeBSDWrite (struct cdev *pDev, struct uio *pUio, int fIo)
395{
396 return 0;
397}
398
399static int VBoxGuestFreeBSDRead (struct cdev *pDev, struct uio *pUio, int fIo)
400{
401 return 0;
402}
403
404static int VBoxGuestFreeBSDDetach(device_t pDevice)
405{
406 struct VBoxGuestDeviceState *pState = device_get_softc(pDevice);
407
408 if (cUsers > 0)
409 return EBUSY;
410
411 /*
412 * Reverse what we did in VBoxGuestFreeBSDAttach.
413 */
414 if (g_VBoxGuestFreeBSDEHTag != NULL)
415 EVENTHANDLER_DEREGISTER(dev_clone, g_VBoxGuestFreeBSDEHTag);
416
417 clone_cleanup(&g_pVBoxGuestFreeBSDClones);
418
419 VBoxGuestFreeBSDRemoveIRQ(pDevice, pState);
420
421 if (pState->pVMMDevMemRes)
422 bus_release_resource(pDevice, SYS_RES_MEMORY, pState->iVMMDevMemResId, pState->pVMMDevMemRes);
423 if (pState->pIOPortRes)
424 bus_release_resource(pDevice, SYS_RES_IOPORT, pState->iIOPortResId, pState->pIOPortRes);
425
426 VBoxGuestDeleteDevExt(&g_DevExt);
427
428 RTR0Term();
429
430 return 0;
431}
432
433/**
434 * Interrupt service routine.
435 *
436 * @returns Whether the interrupt was from VMMDev.
437 * @param pvState Opaque pointer to the device state.
438 */
439static int VBoxGuestFreeBSDISR(void *pvState)
440{
441 LogFlow((DEVICE_NAME ":VBoxGuestFreeBSDISR pvState=%p\n", pvState));
442
443 bool fOurIRQ = VBoxGuestCommonISR(&g_DevExt);
444
445 return fOurIRQ ? 0 : 1;
446}
447
448void VBoxGuestNativeISRMousePollEvent(PVBOXGUESTDEVEXT pDevExt)
449{
450 LogFlow((DEVICE_NAME "::NativeISRMousePollEvent:\n"));
451
452 /*
453 * Wake up poll waiters.
454 */
455 selwakeup(&g_SelInfo);
456}
457
458/**
459 * Sets IRQ for VMMDev.
460 *
461 * @returns FreeBSD error code.
462 * @param pDevice Pointer to the device info structure.
463 * @param pvState Pointer to the state info structure.
464 */
465static int VBoxGuestFreeBSDAddIRQ(device_t pDevice, void *pvState)
466{
467 int iResId = 0;
468 int rc = 0;
469 struct VBoxGuestDeviceState *pState = (struct VBoxGuestDeviceState *)pvState;
470
471 pState->pIrqRes = bus_alloc_resource_any(pDevice, SYS_RES_IRQ, &iResId, RF_SHAREABLE | RF_ACTIVE);
472
473#if __FreeBSD_version >= 700000
474 rc = bus_setup_intr(pDevice, pState->pIrqRes, INTR_TYPE_BIO | INTR_MPSAFE, NULL, (driver_intr_t *)VBoxGuestFreeBSDISR, pState,
475 &pState->pfnIrqHandler);
476#else
477 rc = bus_setup_intr(pDevice, pState->pIrqRes, INTR_TYPE_BIO, (driver_intr_t *)VBoxGuestFreeBSDISR, pState, &pState->pfnIrqHandler);
478#endif
479
480 if (rc)
481 {
482 pState->pfnIrqHandler = NULL;
483 return VERR_DEV_IO_ERROR;
484 }
485
486 pState->iIrqResId = iResId;
487
488 return VINF_SUCCESS;
489}
490
491/**
492 * Removes IRQ for VMMDev.
493 *
494 * @param pDevice Pointer to the device info structure.
495 * @param pvState Opaque pointer to the state info structure.
496 */
497static void VBoxGuestFreeBSDRemoveIRQ(device_t pDevice, void *pvState)
498{
499 struct VBoxGuestDeviceState *pState = (struct VBoxGuestDeviceState *)pvState;
500
501 if (pState->pIrqRes)
502 {
503 bus_teardown_intr(pDevice, pState->pIrqRes, pState->pfnIrqHandler);
504 bus_release_resource(pDevice, SYS_RES_IRQ, 0, pState->pIrqRes);
505 }
506}
507
508static int VBoxGuestFreeBSDAttach(device_t pDevice)
509{
510 int rc = VINF_SUCCESS;
511 int iResId = 0;
512 struct VBoxGuestDeviceState *pState = NULL;
513
514 cUsers = 0;
515
516 /*
517 * Initialize IPRT R0 driver, which internally calls OS-specific r0 init.
518 */
519 rc = RTR0Init(0);
520 if (RT_FAILURE(rc))
521 {
522 LogFunc(("RTR0Init failed.\n"));
523 return ENXIO;
524 }
525
526 pState = device_get_softc(pDevice);
527
528 /*
529 * Allocate I/O port resource.
530 */
531 iResId = PCIR_BAR(0);
532 pState->pIOPortRes = bus_alloc_resource_any(pDevice, SYS_RES_IOPORT, &iResId, RF_ACTIVE);
533 pState->uIOPortBase = rman_get_start(pState->pIOPortRes);
534 pState->iIOPortResId = iResId;
535 if (pState->uIOPortBase)
536 {
537 /*
538 * Map the MMIO region.
539 */
540 iResId = PCIR_BAR(1);
541 pState->pVMMDevMemRes = bus_alloc_resource_any(pDevice, SYS_RES_MEMORY, &iResId, RF_ACTIVE);
542 pState->VMMDevMemHandle = rman_get_bushandle(pState->pVMMDevMemRes);
543 pState->VMMDevMemSize = rman_get_size(pState->pVMMDevMemRes);
544
545 pState->pMMIOBase = rman_get_virtual(pState->pVMMDevMemRes);
546 pState->iVMMDevMemResId = iResId;
547 if (pState->pMMIOBase)
548 {
549 /*
550 * Call the common device extension initializer.
551 */
552 rc = VBoxGuestInitDevExt(&g_DevExt, pState->uIOPortBase,
553 pState->pMMIOBase, pState->VMMDevMemSize,
554 VBOXOSTYPE_FreeBSD, VMMDEV_EVENT_MOUSE_POSITION_CHANGED);
555 if (RT_SUCCESS(rc))
556 {
557 /*
558 * Add IRQ of VMMDev.
559 */
560 rc = VBoxGuestFreeBSDAddIRQ(pDevice, pState);
561 if (RT_SUCCESS(rc))
562 {
563 /*
564 * Configure device cloning.
565 */
566 clone_setup(&g_pVBoxGuestFreeBSDClones);
567 g_VBoxGuestFreeBSDEHTag = EVENTHANDLER_REGISTER(dev_clone, VBoxGuestFreeBSDClone, 0, 1000);
568 if (g_VBoxGuestFreeBSDEHTag)
569 {
570 printf(DEVICE_NAME ": loaded successfully\n");
571 return 0;
572 }
573
574 printf(DEVICE_NAME ": EVENTHANDLER_REGISTER(dev_clone,,,) failed\n");
575 clone_cleanup(&g_pVBoxGuestFreeBSDClones);
576 VBoxGuestFreeBSDRemoveIRQ(pDevice, pState);
577 }
578 else
579 printf((DEVICE_NAME ":VBoxGuestInitDevExt failed.\n"));
580 VBoxGuestDeleteDevExt(&g_DevExt);
581 }
582 else
583 printf((DEVICE_NAME ":VBoxGuestFreeBSDAddIRQ failed.\n"));
584 }
585 else
586 printf((DEVICE_NAME ":MMIO region setup failed.\n"));
587 }
588 else
589 printf((DEVICE_NAME ":IOport setup failed.\n"));
590
591 RTR0Term();
592 return ENXIO;
593}
594
595static int VBoxGuestFreeBSDProbe(device_t pDevice)
596{
597 if ((pci_get_vendor(pDevice) == VMMDEV_VENDORID) && (pci_get_device(pDevice) == VMMDEV_DEVICEID))
598 return 0;
599
600 return ENXIO;
601}
602
603static device_method_t VBoxGuestFreeBSDMethods[] =
604{
605 /* Device interface. */
606 DEVMETHOD(device_probe, VBoxGuestFreeBSDProbe),
607 DEVMETHOD(device_attach, VBoxGuestFreeBSDAttach),
608 DEVMETHOD(device_detach, VBoxGuestFreeBSDDetach),
609 {0,0}
610};
611
612static driver_t VBoxGuestFreeBSDDriver =
613{
614 DEVICE_NAME,
615 VBoxGuestFreeBSDMethods,
616 sizeof(struct VBoxGuestDeviceState),
617};
618
619static devclass_t VBoxGuestFreeBSDClass;
620
621DRIVER_MODULE(vboxguest, pci, VBoxGuestFreeBSDDriver, VBoxGuestFreeBSDClass, 0, 0);
622MODULE_VERSION(vboxguest, 1);
623
624/* Common code that depend on g_DevExt. */
625#include "VBoxGuestIDC-unix.c.h"
626
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