VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuest/VBoxGuest-netbsd.c@ 69206

Last change on this file since 69206 was 63549, checked in by vboxsync, 8 years ago

scm cleanups

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 19.7 KB
Line 
1/* $Id: VBoxGuest-netbsd.c 63549 2016-08-16 12:55:14Z vboxsync $ */
2/** @file
3 * VirtualBox Guest Additions Driver for NetBSD.
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-netbsd.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#include <sys/systm.h>
29#include <sys/select.h>
30#include <sys/conf.h>
31#include <sys/kernel.h>
32#include <sys/kmem.h>
33#include <sys/module.h>
34#include <sys/device.h>
35#include <sys/bus.h>
36#include <sys/poll.h>
37#include <sys/proc.h>
38#include <sys/stat.h>
39#include <sys/selinfo.h>
40#include <sys/queue.h>
41#include <sys/lock.h>
42#include <sys/types.h>
43#include <sys/conf.h>
44#include <sys/malloc.h>
45#include <sys/uio.h>
46#include <sys/file.h>
47#include <sys/filedesc.h>
48#include <sys/vfs_syscalls.h>
49#include <dev/pci/pcivar.h>
50#include <dev/pci/pcireg.h>
51#include <dev/pci/pcidevs.h>
52
53#ifdef PVM
54# undef PVM
55#endif
56#include "VBoxGuestInternal.h"
57#include <VBox/log.h>
58#include <iprt/assert.h>
59#include <iprt/initterm.h>
60#include <iprt/process.h>
61#include <iprt/mem.h>
62#include <iprt/asm.h>
63
64
65/*********************************************************************************************************************************
66* Defined Constants And Macros *
67*********************************************************************************************************************************/
68/** The module name. */
69#define DEVICE_NAME "vboxguest"
70
71
72/*********************************************************************************************************************************
73* Structures and Typedefs *
74*********************************************************************************************************************************/
75typedef struct VBoxGuestDeviceState
76{
77 device_t sc_dev;
78 pci_chipset_tag_t pc;
79 bus_space_tag_t io_tag;
80 bus_space_handle_t io_handle;
81 bus_addr_t uIOPortBase;
82 bus_size_t io_regsize;
83 bus_space_tag_t iVMMDevMemResId;
84 bus_space_handle_t VMMDevMemHandle;
85
86 /** Size of the memory area. */
87 bus_size_t VMMDevMemSize;
88 /** Mapping of the register space */
89 bus_addr_t pMMIOBase;
90
91 /** IRQ resource handle. */
92 pci_intr_handle_t ih;
93 /** Pointer to the IRQ handler. */
94 void *pfnIrqHandler;
95
96 /** Controller features, limits and status. */
97 u_int vboxguest_state;
98} vboxguest_softc;
99
100
101struct vboxguest_session
102{
103 vboxguest_softc *sc;
104 PVBOXGUESTSESSION session;
105};
106
107#define VBOXGUEST_STATE_INITOK 1 << 0
108
109
110/*********************************************************************************************************************************
111* Internal Functions *
112*********************************************************************************************************************************/
113/*
114 * Character device file handlers.
115 */
116static int VBoxGuestNetBSDOpen(dev_t device, int flags, int fmt, struct lwp *process);
117static int VBoxGuestNetBSDClose(struct file *fp);
118static int VBoxGuestNetBSDIOCtl(struct file *fp, u_long cmd, void *addr);
119static int VBoxGuestNetBSDPoll(struct file *fp, int events);
120static void VBoxGuestNetBSDAttach(device_t, device_t, void*);
121static int VBoxGuestNetBSDDetach(device_t pDevice, int flags);
122
123/*
124 * IRQ related functions.
125 */
126static void VBoxGuestNetBSDRemoveIRQ(vboxguest_softc *sc);
127static int VBoxGuestNetBSDAddIRQ(vboxguest_softc *pvState, struct pci_attach_args *pa);
128static int VBoxGuestNetBSDISR(void *pvState);
129
130
131/*********************************************************************************************************************************
132* Global Variables *
133*********************************************************************************************************************************/
134/*
135 * The /dev/vboxguest character device entry points.
136 */
137static struct cdevsw g_VBoxGuestNetBSDChrDevSW =
138{
139 VBoxGuestNetBSDOpen,
140 noclose,
141 noread,
142 nowrite,
143 noioctl,
144 nostop,
145 notty,
146 nopoll,
147 nommap,
148 nokqfilter,
149};
150
151static const struct fileops vboxguest_fileops = {
152 .fo_read = fbadop_read,
153 .fo_write = fbadop_write,
154 .fo_ioctl = VBoxGuestNetBSDIOCtl,
155 .fo_fcntl = fnullop_fcntl,
156 .fo_poll = VBoxGuestNetBSDPoll,
157 .fo_stat = fbadop_stat,
158 .fo_close = VBoxGuestNetBSDClose,
159 .fo_kqfilter = fnullop_kqfilter,
160 .fo_restart = fnullop_restart
161};
162
163/** Device extention & session data association structure. */
164static VBOXGUESTDEVEXT g_DevExt;
165/** Reference counter */
166static volatile uint32_t cUsers;
167/** selinfo structure used for polling. */
168static struct selinfo g_SelInfo;
169
170CFDRIVER_DECL(vboxguest, DV_DULL, NULL);
171extern struct cfdriver vboxguest_cd;
172
173/**
174 * File open handler
175 *
176 */
177static int VBoxGuestNetBSDOpen(dev_t device, int flags, int fmt, struct lwp *process)
178{
179 int rc;
180 vboxguest_softc *vboxguest;
181 struct vboxguest_session *session;
182 file_t *fp;
183 int fd, error;
184
185 LogFlow((DEVICE_NAME ": %s\n", __func__));
186
187 if ((vboxguest = device_lookup_private(&vboxguest_cd, minor(device))) == NULL)
188 {
189 printf("device_lookup_private failed\n");
190 return (ENXIO);
191 }
192
193 if ((vboxguest->vboxguest_state & VBOXGUEST_STATE_INITOK) == 0)
194 {
195 aprint_error_dev(vboxguest->sc_dev, "device not configured\n");
196 return (ENXIO);
197 }
198
199 session = kmem_alloc(sizeof(*session), KM_SLEEP);
200 if (session == NULL)
201 {
202 return (ENOMEM);
203 }
204
205 session->sc = vboxguest;
206
207 if ((error = fd_allocfile(&fp, &fd)) != 0)
208 {
209 kmem_free(session, sizeof(*session));
210 return error;
211 }
212
213 /*
214 * Create a new session.
215 */
216 rc = VGDrvCommonCreateUserSession(&g_DevExt, &session->session);
217 if (! RT_SUCCESS(rc))
218 {
219 aprint_error_dev(vboxguest->sc_dev, "VBox session creation failed\n");
220 closef(fp); /* ??? */
221 kmem_free(session, sizeof(*session));
222 return RTErrConvertToErrno(rc);
223 }
224 ASMAtomicIncU32(&cUsers);
225 return fd_clone(fp, fd, flags, &vboxguest_fileops, session);
226
227}
228
229/**
230 * File close handler
231 *
232 */
233static int VBoxGuestNetBSDClose(struct file *fp)
234{
235 struct vboxguest_session *session = fp->f_data;
236 vboxguest_softc *vboxguest = session->sc;
237
238 LogFlow((DEVICE_NAME ": %s\n", __func__));
239
240 VGDrvCommonCloseSession(&g_DevExt, session->session);
241 ASMAtomicDecU32(&cUsers);
242
243 kmem_free(session, sizeof(*session));
244
245 return 0;
246}
247
248/**
249 * IOCTL handler
250 *
251 */
252static int VBoxGuestNetBSDIOCtl(struct file *fp, u_long command, void *data)
253{
254 struct vboxguest_session *session = fp->f_data;
255 vboxguest_softc *vboxguest = session->sc;
256
257 int rc = 0;
258
259 LogFlow((DEVICE_NAME ": %s: command=%#lx data=%p\n",
260 __func__, command, data));
261
262 /*
263 * Validate the request wrapper.
264 */
265 if (IOCPARM_LEN(command) != sizeof(VBGLBIGREQ))
266 {
267 Log((DEVICE_NAME ": %s: bad request %#lx size=%lu expected=%zu\n",
268 __func__, command, IOCPARM_LEN(command), sizeof(VBGLBIGREQ)));
269 return ENOTTY;
270 }
271
272 PVBGLBIGREQ ReqWrap = (PVBGLBIGREQ)data;
273 if (ReqWrap->u32Magic != VBGLBIGREQ_MAGIC)
274 {
275 Log((DEVICE_NAME ": %s: bad magic %#" PRIx32 "; pArg=%p Cmd=%#lx\n",
276 __func__, ReqWrap->u32Magic, data, command));
277 return EINVAL;
278 }
279 if (RT_UNLIKELY( ReqWrap->cbData == 0
280 || ReqWrap->cbData > _1M*16))
281 {
282 Log((DEVICE_NAME ": %s: bad size %" PRIu32 "; pArg=%p Cmd=%#lx\n",
283 __func__, ReqWrap->cbData, data, command));
284 return EINVAL;
285 }
286
287 /*
288 * Read the request.
289 */
290 void *pvBuf = RTMemTmpAlloc(ReqWrap->cbData);
291 if (RT_UNLIKELY(!pvBuf))
292 {
293 Log((DEVICE_NAME ": %s: RTMemTmpAlloc failed to alloc %" PRIu32 " bytes.\n",
294 __func__, ReqWrap->cbData));
295 return ENOMEM;
296 }
297
298 rc = copyin((void *)(uintptr_t)ReqWrap->pvDataR3, pvBuf, ReqWrap->cbData);
299 if (RT_UNLIKELY(rc))
300 {
301 RTMemTmpFree(pvBuf);
302 Log((DEVICE_NAME ": %s: copyin failed; pvBuf=%p pArg=%p Cmd=%#lx. rc=%d\n",
303 __func__, pvBuf, data, command, rc));
304 aprint_error_dev(vboxguest->sc_dev, "copyin failed\n");
305 return EFAULT;
306 }
307 if (RT_UNLIKELY( ReqWrap->cbData != 0
308 && !VALID_PTR(pvBuf)))
309 {
310 RTMemTmpFree(pvBuf);
311 Log((DEVICE_NAME ": %s: invalid pvBuf=%p\n",
312 __func__, pvBuf));
313 return EINVAL;
314 }
315
316 /*
317 * Process the IOCtl.
318 */
319 size_t cbDataReturned;
320 rc = VGDrvCommonIoCtl(command, &g_DevExt, session->session, pvBuf, ReqWrap->cbData, &cbDataReturned);
321 if (RT_SUCCESS(rc))
322 {
323 rc = 0;
324 if (RT_UNLIKELY(cbDataReturned > ReqWrap->cbData))
325 {
326 Log((DEVICE_NAME ": %s: too much output data %zu expected %" PRIu32 "\n",
327 __func__, cbDataReturned, ReqWrap->cbData));
328 cbDataReturned = ReqWrap->cbData;
329 }
330 if (cbDataReturned > 0)
331 {
332 rc = copyout(pvBuf, (void *)(uintptr_t)ReqWrap->pvDataR3, cbDataReturned);
333 if (RT_UNLIKELY(rc))
334 {
335 Log((DEVICE_NAME ": %s: copyout failed; pvBuf=%p pArg=%p. rc=%d\n",
336 __func__, pvBuf, data, rc));
337 }
338 }
339 } else {
340 Log((DEVICE_NAME ": %s: VGDrvCommonIoCtl failed. rc=%d\n",
341 __func__, rc));
342 rc = -rc;
343 }
344 RTMemTmpFree(pvBuf);
345 return rc;
346}
347
348static int VBoxGuestNetBSDPoll(struct file *fp, int events)
349{
350 struct vboxguest_session *session = fp->f_data;
351 vboxguest_softc *vboxguest = session->sc;
352
353 int rc = 0;
354 int events_processed;
355
356 uint32_t u32CurSeq;
357
358 LogFlow((DEVICE_NAME ": %s\n", __func__));
359
360 u32CurSeq = ASMAtomicUoReadU32(&g_DevExt.u32MousePosChangedSeq);
361 if (session->session->u32MousePosChangedSeq != u32CurSeq)
362 {
363 events_processed = events & (POLLIN | POLLRDNORM);
364 session->session->u32MousePosChangedSeq = u32CurSeq;
365 }
366 else
367 {
368 events_processed = 0;
369
370 selrecord(curlwp, &g_SelInfo);
371 }
372
373 return events_processed;
374}
375
376static int VBoxGuestNetBSDDetach(device_t self, int flags)
377{
378 vboxguest_softc *vboxguest;
379 vboxguest = device_private(self);
380
381 LogFlow((DEVICE_NAME ": %s\n", __func__));
382
383 if (cUsers > 0)
384 return EBUSY;
385
386 if ((vboxguest->vboxguest_state & VBOXGUEST_STATE_INITOK) == 0)
387 return 0;
388
389 /*
390 * Reverse what we did in VBoxGuestNetBSDAttach.
391 */
392
393 VBoxGuestNetBSDRemoveIRQ(vboxguest);
394
395 VGDrvCommonDeleteDevExt(&g_DevExt);
396
397 bus_space_unmap(vboxguest->iVMMDevMemResId, vboxguest->VMMDevMemHandle, vboxguest->VMMDevMemSize);
398 bus_space_unmap(vboxguest->io_tag, vboxguest->io_handle, vboxguest->io_regsize);
399
400 RTR0Term();
401
402 return 0;
403}
404
405/**
406 * Interrupt service routine.
407 *
408 * @returns Whether the interrupt was from VMMDev.
409 * @param pvState Opaque pointer to the device state.
410 */
411static int VBoxGuestNetBSDISR(void *pvState)
412{
413 LogFlow((DEVICE_NAME ": %s: pvState=%p\n", __func__, pvState));
414
415 bool fOurIRQ = VGDrvCommonISR(&g_DevExt);
416
417 return fOurIRQ ? 0 : 1;
418}
419
420void VGDrvNativeISRMousePollEvent(PVBOXGUESTDEVEXT pDevExt)
421{
422 LogFlow((DEVICE_NAME ": %s\n", __func__));
423
424 /*
425 * Wake up poll waiters.
426 */
427 selnotify(&g_SelInfo, 0, 0);
428}
429
430/**
431 * Sets IRQ for VMMDev.
432 *
433 * @returns NetBSD error code.
434 * @param vboxguest Pointer to the state info structure.
435 * @param pa Pointer to the PCI attach arguments.
436 */
437static int VBoxGuestNetBSDAddIRQ(vboxguest_softc *vboxguest, struct pci_attach_args *pa)
438{
439 int iResId = 0;
440 int rc = 0;
441 const char *intrstr;
442#if __NetBSD_Prereq__(6, 99, 39)
443 char intstrbuf[100];
444#endif
445
446 LogFlow((DEVICE_NAME ": %s\n", __func__));
447
448 if (pci_intr_map(pa, &vboxguest->ih))
449 {
450 aprint_error_dev(vboxguest->sc_dev, "couldn't map interrupt.\n");
451 return VERR_DEV_IO_ERROR;
452 }
453
454 intrstr = pci_intr_string(vboxguest->pc, vboxguest->ih
455#if __NetBSD_Prereq__(6, 99, 39)
456 , intstrbuf, sizeof(intstrbuf)
457#endif
458 );
459 aprint_normal_dev(vboxguest->sc_dev, "interrupting at %s\n", intrstr);
460
461 vboxguest->pfnIrqHandler = pci_intr_establish(vboxguest->pc, vboxguest->ih, IPL_BIO, VBoxGuestNetBSDISR, vboxguest);
462 if (vboxguest->pfnIrqHandler == NULL)
463 {
464 aprint_error_dev(vboxguest->sc_dev, "couldn't establish interrupt\n");
465 return VERR_DEV_IO_ERROR;
466 }
467
468 return VINF_SUCCESS;
469}
470
471/**
472 * Removes IRQ for VMMDev.
473 *
474 * @param vboxguest Opaque pointer to the state info structure.
475 */
476static void VBoxGuestNetBSDRemoveIRQ(vboxguest_softc *vboxguest)
477{
478 LogFlow((DEVICE_NAME ": %s\n", __func__));
479
480 if (vboxguest->pfnIrqHandler)
481 {
482 pci_intr_disestablish(vboxguest->pc, vboxguest->pfnIrqHandler);
483 }
484}
485
486static void VBoxGuestNetBSDAttach(device_t parent, device_t self, void *aux)
487{
488 int rc = VINF_SUCCESS;
489 int iResId = 0;
490 vboxguest_softc *vboxguest;
491 struct pci_attach_args *pa = aux;
492 bus_space_tag_t iot, memt;
493 bus_space_handle_t ioh, memh;
494 bus_dma_segment_t seg;
495 int ioh_valid, memh_valid;
496
497 cUsers = 0;
498
499 aprint_normal(": VirtualBox Guest\n");
500
501 vboxguest = device_private(self);
502 vboxguest->sc_dev = self;
503
504 /*
505 * Initialize IPRT R0 driver, which internally calls OS-specific r0 init.
506 */
507 rc = RTR0Init(0);
508 if (RT_FAILURE(rc))
509 {
510 LogFunc(("RTR0Init failed.\n"));
511 aprint_error_dev(vboxguest->sc_dev, "RTR0Init failed\n");
512 return;
513 }
514
515 vboxguest->pc = pa->pa_pc;
516
517 /*
518 * Allocate I/O port resource.
519 */
520 ioh_valid = (pci_mapreg_map(pa, PCI_MAPREG_START, PCI_MAPREG_TYPE_IO, 0, &vboxguest->io_tag, &vboxguest->io_handle, &vboxguest->uIOPortBase, &vboxguest->io_regsize) == 0);
521
522 if (ioh_valid)
523 {
524
525 /*
526 * Map the MMIO region.
527 */
528 memh_valid = (pci_mapreg_map(pa, PCI_MAPREG_START+4, PCI_MAPREG_TYPE_MEM, BUS_SPACE_MAP_LINEAR, &vboxguest->iVMMDevMemResId, &vboxguest->VMMDevMemHandle, &vboxguest->pMMIOBase, &vboxguest->VMMDevMemSize) == 0);
529 if (memh_valid)
530 {
531 /*
532 * Call the common device extension initializer.
533 */
534 rc = VGDrvCommonInitDevExt(&g_DevExt, vboxguest->uIOPortBase,
535 bus_space_vaddr(vboxguest->iVMMDevMemResId,
536 vboxguest->VMMDevMemHandle),
537 vboxguest->VMMDevMemSize,
538#if ARCH_BITS == 64
539 VBOXOSTYPE_NetBSD_x64,
540#else
541 VBOXOSTYPE_NetBSD,
542#endif
543 VMMDEV_EVENT_MOUSE_POSITION_CHANGED);
544 if (RT_SUCCESS(rc))
545 {
546 /*
547 * Add IRQ of VMMDev.
548 */
549 rc = VBoxGuestNetBSDAddIRQ(vboxguest, pa);
550 if (RT_SUCCESS(rc))
551 {
552 vboxguest->vboxguest_state |= VBOXGUEST_STATE_INITOK;
553 return;
554 }
555 VGDrvCommonDeleteDevExt(&g_DevExt);
556 }
557 else
558 {
559 aprint_error_dev(vboxguest->sc_dev, "init failed\n");
560 }
561 bus_space_unmap(vboxguest->iVMMDevMemResId, vboxguest->VMMDevMemHandle, vboxguest->VMMDevMemSize);
562 }
563 else
564 {
565 aprint_error_dev(vboxguest->sc_dev, "MMIO mapping failed\n");
566 }
567 bus_space_unmap(vboxguest->io_tag, vboxguest->io_handle, vboxguest->io_regsize);
568 }
569 else
570 {
571 aprint_error_dev(vboxguest->sc_dev, "IO mapping failed\n");
572 }
573
574 RTR0Term();
575 return;
576}
577
578static int
579VBoxGuestNetBSDMatch(device_t parent, cfdata_t match, void *aux)
580{
581 const struct pci_attach_args *pa = aux;
582
583 if ( PCI_VENDOR(pa->pa_id) == VMMDEV_VENDORID
584 && PCI_PRODUCT(pa->pa_id) == VMMDEV_DEVICEID)
585 {
586 return 1;
587 }
588
589 return 0;
590}
591
592/* Common code that depend on g_DevExt. */
593#include "VBoxGuestIDC-unix.c.h"
594
595CFATTACH_DECL_NEW(vboxguest, sizeof(vboxguest_softc),
596 VBoxGuestNetBSDMatch, VBoxGuestNetBSDAttach, VBoxGuestNetBSDDetach, NULL);
597
598MODULE(MODULE_CLASS_DRIVER, vboxguest, "pci");
599
600static int loc[2] = {-1, -1};
601
602static const struct cfparent pspec = {
603 "pci", "pci", DVUNIT_ANY
604};
605
606static struct cfdata vboxguest_cfdata[] = {
607 {
608 .cf_name = "vboxguest",
609 .cf_atname = "vboxguest",
610 .cf_unit = 0, /* Only unit 0 is ever used */
611 .cf_fstate = FSTATE_STAR,
612 .cf_loc = loc,
613 .cf_flags = 0,
614 .cf_pspec = &pspec,
615 },
616 { NULL, NULL, 0, 0, NULL, 0, NULL }
617};
618
619static int
620vboxguest_modcmd(modcmd_t cmd, void *opaque)
621{
622 devmajor_t bmajor, cmajor;
623 int error;
624 register_t retval;
625
626 LogFlow((DEVICE_NAME ": %s\n", __func__));
627
628 bmajor = cmajor = NODEVMAJOR;
629 switch (cmd)
630 {
631 case MODULE_CMD_INIT:
632 error = config_cfdriver_attach(&vboxguest_cd);
633 if (error)
634 {
635 printf("config_cfdriver_attach failed: %d", error);
636 break;
637 }
638 error = config_cfattach_attach(vboxguest_cd.cd_name, &vboxguest_ca);
639 if (error)
640 {
641 config_cfdriver_detach(&vboxguest_cd);
642 printf("%s: unable to register cfattach\n", vboxguest_cd.cd_name);
643 break;
644 }
645 error = config_cfdata_attach(vboxguest_cfdata, 1);
646 if (error)
647 {
648 printf("%s: unable to attach cfdata\n", vboxguest_cd.cd_name);
649 config_cfattach_detach(vboxguest_cd.cd_name, &vboxguest_ca);
650 config_cfdriver_detach(&vboxguest_cd);
651 break;
652 }
653
654 error = devsw_attach("vboxguest", NULL, &bmajor, &g_VBoxGuestNetBSDChrDevSW, &cmajor);
655
656 if (error == EEXIST)
657 error = 0; /* maybe built-in ... improve eventually */
658
659 if (error)
660 break;
661
662 error = do_sys_mknod(curlwp, "/dev/vboxguest", 0666|S_IFCHR, makedev(cmajor, 0), &retval, UIO_SYSSPACE);
663 if (error == EEXIST)
664 error = 0;
665 break;
666
667 case MODULE_CMD_FINI:
668 error = config_cfdata_detach(vboxguest_cfdata);
669 if (error)
670 break;
671 error = config_cfattach_detach(vboxguest_cd.cd_name, &vboxguest_ca);
672 if (error)
673 break;
674 config_cfdriver_detach(&vboxguest_cd);
675 error = devsw_detach(NULL, &g_VBoxGuestNetBSDChrDevSW);
676 break;
677
678 default:
679 return ENOTTY;
680 }
681 return error;
682}
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