VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuest/VBoxGuest-linux.c@ 42738

Last change on this file since 42738 was 41722, checked in by vboxsync, 13 years ago

Additions/common/VBoxGuest: fixed copyright dates and licence headers and removed a copyright text that is not relevant to this code.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 33.0 KB
Line 
1/* $Rev: 41722 $ */
2/** @file
3 * VBoxGuest - Linux specifics.
4 *
5 * Note. Unfortunately, the difference between this and SUPDrv-linux.c is
6 * a little bit too big to be helpful.
7 */
8
9/*
10 * Copyright (C) 2006-2012 Oracle Corporation
11 *
12 * This file is part of VirtualBox Open Source Edition (OSE), as
13 * available from http://www.virtualbox.org. This file is free software;
14 * you can redistribute it and/or modify it under the terms of the GNU
15 * General Public License (GPL) as published by the Free Software
16 * Foundation, in version 2 as it comes in the "COPYING" file of the
17 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
18 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
19 */
20
21/*******************************************************************************
22* Header Files *
23*******************************************************************************/
24#define LOG_GROUP LOG_GROUP_SUP_DRV
25
26#include "the-linux-kernel.h"
27
28#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 15)
29# define VBOXGUEST_WITH_INPUT_DRIVER
30#endif
31
32#include "VBoxGuestInternal.h"
33#ifdef VBOXGUEST_WITH_INPUT_DRIVER
34# include <linux/input.h>
35#endif
36#include <linux/miscdevice.h>
37#include <linux/poll.h>
38#include <VBox/version.h>
39
40#include <iprt/assert.h>
41#include <iprt/asm.h>
42#include <iprt/err.h>
43#include <iprt/initterm.h>
44#include <iprt/mem.h>
45#include <iprt/mp.h>
46#include <iprt/process.h>
47#include <iprt/spinlock.h>
48#include <iprt/semaphore.h>
49#include <VBox/log.h>
50
51
52/*******************************************************************************
53* Defined Constants And Macros *
54*******************************************************************************/
55/** The device name. */
56#define DEVICE_NAME "vboxguest"
57/** The device name for the device node open to everyone.. */
58#define DEVICE_NAME_USER "vboxuser"
59/** The name of the PCI driver */
60#define DRIVER_NAME DEVICE_NAME
61
62
63/* 2.4.x compatibility macros that may or may not be defined. */
64#ifndef IRQ_RETVAL
65# define irqreturn_t void
66# define IRQ_RETVAL(n)
67#endif
68
69
70/*******************************************************************************
71* Internal Functions *
72*******************************************************************************/
73static void vboxguestLinuxTermPci(struct pci_dev *pPciDev);
74static int vboxguestLinuxModInit(void);
75static void vboxguestLinuxModExit(void);
76static int vboxguestLinuxOpen(struct inode *pInode, struct file *pFilp);
77static int vboxguestLinuxRelease(struct inode *pInode, struct file *pFilp);
78#ifdef HAVE_UNLOCKED_IOCTL
79static long vboxguestLinuxIOCtl(struct file *pFilp, unsigned int uCmd, unsigned long ulArg);
80#else
81static int vboxguestLinuxIOCtl(struct inode *pInode, struct file *pFilp, unsigned int uCmd, unsigned long ulArg);
82#endif
83static int vboxguestFAsync(int fd, struct file *pFile, int fOn);
84static unsigned int vboxguestPoll(struct file *pFile, poll_table *pPt);
85static ssize_t vboxguestRead(struct file *pFile, char *pbBuf, size_t cbRead, loff_t *poff);
86
87
88/*******************************************************************************
89* Global Variables *
90*******************************************************************************/
91/**
92 * Device extention & session data association structure.
93 */
94static VBOXGUESTDEVEXT g_DevExt;
95/** The PCI device. */
96static struct pci_dev *g_pPciDev = NULL;
97/** The base of the I/O port range. */
98static RTIOPORT g_IOPortBase;
99/** The base of the MMIO range. */
100static RTHCPHYS g_MMIOPhysAddr = NIL_RTHCPHYS;
101/** The size of the MMIO range as seen by PCI. */
102static uint32_t g_cbMMIO;
103/** The pointer to the mapping of the MMIO range. */
104static void *g_pvMMIOBase;
105/** Wait queue used by polling. */
106static wait_queue_head_t g_PollEventQueue;
107/** Asynchronous notification stuff. */
108static struct fasync_struct *g_pFAsyncQueue;
109#ifdef VBOXGUEST_WITH_INPUT_DRIVER
110/** Pre-allocated mouse status VMMDev request for use in the IRQ
111 * handler. */
112static VMMDevReqMouseStatus *g_pMouseStatusReq;
113#endif
114#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0)
115/** Whether we've create the logger or not. */
116static volatile bool g_fLoggerCreated;
117/** Release logger group settings. */
118static char g_szLogGrp[128];
119/** Release logger flags settings. */
120static char g_szLogFlags[128];
121/** Release logger destination settings. */
122static char g_szLogDst[128];
123# if 0
124/** Debug logger group settings. */
125static char g_szDbgLogGrp[128];
126/** Debug logger flags settings. */
127static char g_szDbgLogFlags[128];
128/** Debug logger destination settings. */
129static char g_szDbgLogDst[128];
130# endif
131#endif
132
133/** Our file node major id.
134 * Either set dynamically at run time or statically at compile time. */
135#ifdef CONFIG_VBOXGUEST_MAJOR
136static unsigned int g_iModuleMajor = CONFIG_VBOXGUEST_MAJOR;
137#else
138static unsigned int g_iModuleMajor = 0;
139#endif
140#ifdef CONFIG_VBOXADD_MAJOR
141# error "CONFIG_VBOXADD_MAJOR -> CONFIG_VBOXGUEST_MAJOR"
142#endif
143
144/** The input device handle */
145#ifdef VBOXGUEST_WITH_INPUT_DRIVER
146static struct input_dev *g_pInputDevice = NULL;
147#endif
148
149/** The file_operations structure. */
150static struct file_operations g_FileOps =
151{
152 owner: THIS_MODULE,
153 open: vboxguestLinuxOpen,
154 release: vboxguestLinuxRelease,
155#ifdef HAVE_UNLOCKED_IOCTL
156 unlocked_ioctl: vboxguestLinuxIOCtl,
157#else
158 ioctl: vboxguestLinuxIOCtl,
159#endif
160 fasync: vboxguestFAsync,
161 read: vboxguestRead,
162 poll: vboxguestPoll,
163 llseek: no_llseek,
164};
165
166/** The miscdevice structure. */
167static struct miscdevice g_MiscDevice =
168{
169 minor: MISC_DYNAMIC_MINOR,
170 name: DEVICE_NAME,
171 fops: &g_FileOps,
172};
173
174/** The file_operations structure for the user device.
175 * @remarks For the time being we'll be using the same implementation as
176 * /dev/vboxguest here. */
177static struct file_operations g_FileOpsUser =
178{
179 owner: THIS_MODULE,
180 open: vboxguestLinuxOpen,
181 release: vboxguestLinuxRelease,
182#ifdef HAVE_UNLOCKED_IOCTL
183 unlocked_ioctl: vboxguestLinuxIOCtl,
184#else
185 ioctl: vboxguestLinuxIOCtl,
186#endif
187};
188
189/** The miscdevice structure for the user device. */
190static struct miscdevice g_MiscDeviceUser =
191{
192 minor: MISC_DYNAMIC_MINOR,
193 name: DEVICE_NAME_USER,
194 fops: &g_FileOpsUser,
195};
196
197
198/** PCI hotplug structure. */
199static const struct pci_device_id __devinitdata g_VBoxGuestPciId[] =
200{
201 {
202 vendor: VMMDEV_VENDORID,
203 device: VMMDEV_DEVICEID
204 },
205 {
206 /* empty entry */
207 }
208};
209MODULE_DEVICE_TABLE(pci, g_VBoxGuestPciId);
210
211static PVBOXGUESTSESSION g_pKernelSession = NULL;
212
213/**
214 * Converts a VBox status code to a linux error code.
215 *
216 * @returns corresponding negative linux error code.
217 * @param rc supdrv error code (SUPDRV_ERR_* defines).
218 */
219static int vboxguestLinuxConvertToNegErrno(int rc)
220{
221 if ( rc > -1000
222 && rc < 1000)
223 return -RTErrConvertToErrno(rc);
224 switch (rc)
225 {
226 case VERR_HGCM_SERVICE_NOT_FOUND: return -ESRCH;
227 case VINF_HGCM_CLIENT_REJECTED: return 0;
228 case VERR_HGCM_INVALID_CMD_ADDRESS: return -EFAULT;
229 case VINF_HGCM_ASYNC_EXECUTE: return 0;
230 case VERR_HGCM_INTERNAL: return -EPROTO;
231 case VERR_HGCM_INVALID_CLIENT_ID: return -EINVAL;
232 case VINF_HGCM_SAVE_STATE: return 0;
233 /* No reason to return this to a guest */
234 // case VERR_HGCM_SERVICE_EXISTS: return -EEXIST;
235 default:
236 AssertMsgFailed(("Unhandled error code %Rrc\n", rc));
237 return -EPROTO;
238 }
239}
240
241
242
243/**
244 * Does the PCI detection and init of the device.
245 *
246 * @returns 0 on success, negated errno on failure.
247 */
248static int vboxguestLinuxProbePci(struct pci_dev *pPciDev,
249 const struct pci_device_id *id)
250{
251 int rc;
252
253 NOREF(id);
254 AssertReturn(!g_pPciDev, -EINVAL);
255 rc = pci_enable_device(pPciDev);
256 if (rc >= 0)
257 {
258 /* I/O Ports are mandatory, the MMIO bit is not. */
259 g_IOPortBase = pci_resource_start(pPciDev, 0);
260 if (g_IOPortBase != 0)
261 {
262 /*
263 * Map the register address space.
264 */
265 g_MMIOPhysAddr = pci_resource_start(pPciDev, 1);
266 g_cbMMIO = pci_resource_len(pPciDev, 1);
267 if (request_mem_region(g_MMIOPhysAddr, g_cbMMIO, DEVICE_NAME) != NULL)
268 {
269 g_pvMMIOBase = ioremap(g_MMIOPhysAddr, g_cbMMIO);
270 if (g_pvMMIOBase)
271 {
272 /** @todo why aren't we requesting ownership of the I/O ports as well? */
273 g_pPciDev = pPciDev;
274 return 0;
275 }
276
277 /* failure cleanup path */
278 LogRel((DEVICE_NAME ": ioremap failed; MMIO Addr=%RHp cb=%#x\n", g_MMIOPhysAddr, g_cbMMIO));
279 rc = -ENOMEM;
280 release_mem_region(g_MMIOPhysAddr, g_cbMMIO);
281 }
282 else
283 {
284 LogRel((DEVICE_NAME ": failed to obtain adapter memory\n"));
285 rc = -EBUSY;
286 }
287 g_MMIOPhysAddr = NIL_RTHCPHYS;
288 g_cbMMIO = 0;
289 g_IOPortBase = 0;
290 }
291 else
292 {
293 LogRel((DEVICE_NAME ": did not find expected hardware resources\n"));
294 rc = -ENXIO;
295 }
296 pci_disable_device(pPciDev);
297 }
298 else
299 LogRel((DEVICE_NAME ": could not enable device: %d\n", rc));
300 return rc;
301}
302
303
304/**
305 * Clean up the usage of the PCI device.
306 */
307static void vboxguestLinuxTermPci(struct pci_dev *pPciDev)
308{
309 g_pPciDev = NULL;
310 if (pPciDev)
311 {
312 iounmap(g_pvMMIOBase);
313 g_pvMMIOBase = NULL;
314
315 release_mem_region(g_MMIOPhysAddr, g_cbMMIO);
316 g_MMIOPhysAddr = NIL_RTHCPHYS;
317 g_cbMMIO = 0;
318
319 pci_disable_device(pPciDev);
320 }
321}
322
323
324/** Structure for registering the PCI driver. */
325static struct pci_driver g_PciDriver =
326{
327 name: DRIVER_NAME,
328 id_table: g_VBoxGuestPciId,
329 probe: vboxguestLinuxProbePci,
330 remove: vboxguestLinuxTermPci
331};
332
333
334/**
335 * Interrupt service routine.
336 *
337 * @returns In 2.4 it returns void.
338 * In 2.6 we indicate whether we've handled the IRQ or not.
339 *
340 * @param iIrq The IRQ number.
341 * @param pvDevId The device ID, a pointer to g_DevExt.
342 * @param pvRegs Register set. Removed in 2.6.19.
343 */
344#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 19)
345static irqreturn_t vboxguestLinuxISR(int iIrrq, void *pvDevId)
346#else
347static irqreturn_t vboxguestLinuxISR(int iIrrq, void *pvDevId, struct pt_regs *pRegs)
348#endif
349{
350 bool fTaken = VBoxGuestCommonISR(&g_DevExt);
351 return IRQ_RETVAL(fTaken);
352}
353
354
355/**
356 * Registers the ISR and initializes the poll wait queue.
357 */
358static int __init vboxguestLinuxInitISR(void)
359{
360 int rc;
361
362 init_waitqueue_head(&g_PollEventQueue);
363 rc = request_irq(g_pPciDev->irq,
364 vboxguestLinuxISR,
365#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 20)
366 IRQF_SHARED,
367#else
368 SA_SHIRQ,
369#endif
370 DEVICE_NAME,
371 &g_DevExt);
372 if (rc)
373 {
374 LogRel((DEVICE_NAME ": could not request IRQ %d: err=%d\n", g_pPciDev->irq, rc));
375 return rc;
376 }
377 return 0;
378}
379
380
381/**
382 * Deregisters the ISR.
383 */
384static void vboxguestLinuxTermISR(void)
385{
386 free_irq(g_pPciDev->irq, &g_DevExt);
387}
388
389
390#ifdef VBOXGUEST_WITH_INPUT_DRIVER
391/** Calls the kernel IOCtl to report mouse status to the host on behalf of
392 * our kernel session. */
393static int vboxguestLinuxSetMouseStatus(uint32_t fStatus)
394{
395 return VBoxGuestCommonIOCtl(VBOXGUEST_IOCTL_SET_MOUSE_STATUS, &g_DevExt,
396 g_pKernelSession, &fStatus, sizeof(fStatus),
397 NULL);
398}
399
400
401/** Called when the input device is first opened. Sets up absolute reporting.
402 */
403static int vboxguestOpenInputDevice(struct input_dev *pDev)
404{
405 NOREF(pDev);
406 if (RT_FAILURE(vboxguestLinuxSetMouseStatus
407 ( VMMDEV_MOUSE_GUEST_CAN_ABSOLUTE
408 | VMMDEV_MOUSE_NEW_PROTOCOL)))
409 return ENODEV;
410 return 0;
411}
412
413
414/** Called if all open handles to the device are closed, disables absolute
415 * reporting. */
416static void vboxguestCloseInputDevice(struct input_dev *pDev)
417{
418 NOREF(pDev);
419 vboxguestLinuxSetMouseStatus(0);
420}
421
422
423/**
424 * Creates the kernel input device.
425 */
426static int __init vboxguestLinuxCreateInputDevice(void)
427{
428 int rc;
429
430 rc = VbglGRAlloc((VMMDevRequestHeader **)&g_pMouseStatusReq,
431 sizeof(*g_pMouseStatusReq),
432 VMMDevReq_GetMouseStatus);
433 if (RT_FAILURE(rc))
434 return -ENOMEM;
435 g_pInputDevice = input_allocate_device();
436 if (!g_pInputDevice)
437 {
438 VbglGRFree(&g_pMouseStatusReq->header);
439 return -ENOMEM;
440 }
441 g_pInputDevice->id.bustype = BUS_PCI;
442 g_pInputDevice->id.vendor = VMMDEV_VENDORID;
443 g_pInputDevice->id.product = VMMDEV_DEVICEID;
444 g_pInputDevice->id.version = VBOX_SHORT_VERSION;
445 g_pInputDevice->open = vboxguestOpenInputDevice;
446 g_pInputDevice->close = vboxguestCloseInputDevice;
447# if LINUX_VERSION_CODE < KERNEL_VERSION(2, 6, 22)
448 g_pInputDevice->cdev.dev = &g_pPciDev->dev;
449# else
450 g_pInputDevice->dev.parent = &g_pPciDev->dev;
451# endif
452 {
453 int rc = input_register_device(g_pInputDevice);
454 if (rc)
455 {
456 VbglGRFree(&g_pMouseStatusReq->header);
457 input_free_device(g_pInputDevice);
458 return rc;
459 }
460 }
461 /* Do what one of our competitors apparently does as that works. */
462 ASMBitSet(g_pInputDevice->evbit, EV_ABS);
463 ASMBitSet(g_pInputDevice->evbit, EV_KEY);
464# ifdef EV_SYN
465 ASMBitSet(g_pInputDevice->evbit, EV_SYN);
466# endif
467 input_set_abs_params(g_pInputDevice, ABS_X, VMMDEV_MOUSE_RANGE_MIN,
468 VMMDEV_MOUSE_RANGE_MAX, 0, 0);
469 input_set_abs_params(g_pInputDevice, ABS_Y, VMMDEV_MOUSE_RANGE_MIN,
470 VMMDEV_MOUSE_RANGE_MAX, 0, 0);
471 ASMBitSet(g_pInputDevice->keybit, BTN_MOUSE);
472 /** @todo this string should be in a header file somewhere. */
473 g_pInputDevice->name = "VirtualBox mouse integration";
474 return 0;
475}
476
477
478/**
479 * Terminates the kernel input device.
480 */
481static void vboxguestLinuxTermInputDevice(void)
482{
483 VbglGRFree(&g_pMouseStatusReq->header);
484 input_unregister_device(g_pInputDevice);
485 input_free_device(g_pInputDevice);
486}
487#endif
488
489
490/**
491 * Creates the device nodes.
492 *
493 * @returns 0 on success, negated errno on failure.
494 */
495static int __init vboxguestLinuxInitDeviceNodes(void)
496{
497 int rc;
498
499 /*
500 * The full feature device node.
501 */
502 if (g_iModuleMajor > 0)
503 {
504 rc = register_chrdev(g_iModuleMajor, DEVICE_NAME, &g_FileOps);
505 if (rc < 0)
506 {
507 LogRel((DEVICE_NAME ": register_chrdev failed: g_iModuleMajor: %d, rc: %d\n", g_iModuleMajor, rc));
508 return rc;
509 }
510 }
511 else
512 {
513 rc = misc_register(&g_MiscDevice);
514 if (rc)
515 {
516 LogRel((DEVICE_NAME ": misc_register failed for %s (rc=%d)\n", DEVICE_NAME, rc));
517 return rc;
518 }
519 }
520
521 /*
522 * The device node intended to be accessible by all users.
523 */
524 rc = misc_register(&g_MiscDeviceUser);
525 if (rc)
526 {
527 LogRel((DEVICE_NAME ": misc_register failed for %s (rc=%d)\n", DEVICE_NAME_USER, rc));
528 if (g_iModuleMajor > 0)
529 unregister_chrdev(g_iModuleMajor, DEVICE_NAME);
530 else
531 misc_deregister(&g_MiscDevice);
532 return rc;
533 }
534
535 return 0;
536}
537
538
539/**
540 * Deregisters the device nodes.
541 */
542static void vboxguestLinuxTermDeviceNodes(void)
543{
544 if (g_iModuleMajor > 0)
545 unregister_chrdev(g_iModuleMajor, DEVICE_NAME);
546 else
547 misc_deregister(&g_MiscDevice);
548 misc_deregister(&g_MiscDeviceUser);
549}
550
551
552/**
553 * Initialize module.
554 *
555 * @returns appropriate status code.
556 */
557static int __init vboxguestLinuxModInit(void)
558{
559 static const char * const s_apszGroups[] = VBOX_LOGGROUP_NAMES;
560 PRTLOGGER pRelLogger;
561 int rc;
562
563 /*
564 * Initialize IPRT first.
565 */
566 rc = RTR0Init(0);
567 if (RT_FAILURE(rc))
568 {
569 printk(KERN_ERR DEVICE_NAME ": RTR0Init failed, rc=%d.\n", rc);
570 return -EINVAL;
571 }
572
573 /*
574 * Create the release log.
575 * (We do that here instead of common code because we want to log
576 * early failures using the LogRel macro.)
577 */
578 rc = RTLogCreate(&pRelLogger, 0 /* fFlags */, "all",
579 "VBOX_RELEASE_LOG", RT_ELEMENTS(s_apszGroups), s_apszGroups,
580 RTLOGDEST_STDOUT | RTLOGDEST_DEBUGGER | RTLOGDEST_USER, NULL);
581 if (RT_SUCCESS(rc))
582 {
583#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0)
584 RTLogGroupSettings(pRelLogger, g_szLogGrp);
585 RTLogFlags(pRelLogger, g_szLogFlags);
586 RTLogDestinations(pRelLogger, g_szLogDst);
587#endif
588 RTLogRelSetDefaultInstance(pRelLogger);
589 }
590#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0)
591 g_fLoggerCreated = true;
592#endif
593
594 /*
595 * Locate and initialize the PCI device.
596 */
597 rc = pci_register_driver(&g_PciDriver);
598 if (rc >= 0 && g_pPciDev)
599 {
600 /*
601 * Register the interrupt service routine for it.
602 */
603 rc = vboxguestLinuxInitISR();
604 if (rc >= 0)
605 {
606 /*
607 * Call the common device extension initializer.
608 */
609#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0) && defined(RT_ARCH_X86)
610 VBOXOSTYPE enmOSType = VBOXOSTYPE_Linux26;
611#elif LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0) && defined(RT_ARCH_AMD64)
612 VBOXOSTYPE enmOSType = VBOXOSTYPE_Linux26_x64;
613#elif LINUX_VERSION_CODE >= KERNEL_VERSION(2, 4, 0) && defined(RT_ARCH_X86)
614 VBOXOSTYPE enmOSType = VBOXOSTYPE_Linux24;
615#elif LINUX_VERSION_CODE >= KERNEL_VERSION(2, 4, 0) && defined(RT_ARCH_AMD64)
616 VBOXOSTYPE enmOSType = VBOXOSTYPE_Linux24_x64;
617#else
618# warning "huh? which arch + version is this?"
619 VBOXOSTYPE enmOsType = VBOXOSTYPE_Linux;
620#endif
621 rc = VBoxGuestInitDevExt(&g_DevExt,
622 g_IOPortBase,
623 g_pvMMIOBase,
624 g_cbMMIO,
625 enmOSType,
626 VMMDEV_EVENT_MOUSE_POSITION_CHANGED);
627 if (RT_SUCCESS(rc))
628 {
629 /*
630 * Create the kernel session for this driver.
631 */
632 rc = VBoxGuestCreateKernelSession(&g_DevExt,
633 &g_pKernelSession);
634 if (RT_SUCCESS(rc))
635 {
636 /*
637 * Create the kernel input device.
638 */
639#ifdef VBOXGUEST_WITH_INPUT_DRIVER
640 rc = vboxguestLinuxCreateInputDevice();
641 if (rc >= 0)
642 {
643#endif
644 /*
645 * Finally, create the device nodes.
646 */
647 rc = vboxguestLinuxInitDeviceNodes();
648 if (rc >= 0)
649 {
650 /* some useful information for the user but don't show this on the console */
651 LogRel((DEVICE_NAME ": major %d, IRQ %d, I/O port %RTiop, MMIO at %RHp (size 0x%x)\n",
652 g_iModuleMajor, g_pPciDev->irq, g_IOPortBase, g_MMIOPhysAddr, g_cbMMIO));
653 printk(KERN_DEBUG DEVICE_NAME ": Successfully loaded version "
654 VBOX_VERSION_STRING " (interface " RT_XSTR(VMMDEV_VERSION) ")\n");
655 return rc;
656 }
657
658 /* bail out */
659#ifdef VBOXGUEST_WITH_INPUT_DRIVER
660 vboxguestLinuxTermInputDevice();
661 }
662 else
663 {
664 LogRel((DEVICE_NAME ": vboxguestCreateInputDevice failed with rc=%Rrc\n", rc));
665 rc = RTErrConvertFromErrno(rc);
666 }
667#endif
668 VBoxGuestCloseSession(&g_DevExt, g_pKernelSession);
669 }
670 VBoxGuestDeleteDevExt(&g_DevExt);
671 }
672 else
673 {
674 LogRel((DEVICE_NAME ": VBoxGuestInitDevExt failed with rc=%Rrc\n", rc));
675 rc = RTErrConvertFromErrno(rc);
676 }
677 vboxguestLinuxTermISR();
678 }
679 }
680 else
681 {
682 LogRel((DEVICE_NAME ": PCI device not found, probably running on physical hardware.\n"));
683 rc = -ENODEV;
684 }
685 pci_unregister_driver(&g_PciDriver);
686 RTLogDestroy(RTLogRelSetDefaultInstance(NULL));
687 RTLogDestroy(RTLogSetDefaultInstance(NULL));
688 RTR0Term();
689 return rc;
690}
691
692
693/**
694 * Unload the module.
695 */
696static void __exit vboxguestLinuxModExit(void)
697{
698 /*
699 * Inverse order of init.
700 */
701 vboxguestLinuxTermDeviceNodes();
702#ifdef VBOXGUEST_WITH_INPUT_DRIVER
703 vboxguestLinuxTermInputDevice();
704#endif
705 VBoxGuestCloseSession(&g_DevExt, g_pKernelSession);
706 VBoxGuestDeleteDevExt(&g_DevExt);
707 vboxguestLinuxTermISR();
708 pci_unregister_driver(&g_PciDriver);
709 RTLogDestroy(RTLogRelSetDefaultInstance(NULL));
710 RTLogDestroy(RTLogSetDefaultInstance(NULL));
711 RTR0Term();
712}
713
714
715/**
716 * Device open. Called on open /dev/vboxdrv
717 *
718 * @param pInode Pointer to inode info structure.
719 * @param pFilp Associated file pointer.
720 */
721static int vboxguestLinuxOpen(struct inode *pInode, struct file *pFilp)
722{
723 int rc;
724 PVBOXGUESTSESSION pSession;
725 Log((DEVICE_NAME ": pFilp=%p pid=%d/%d %s\n", pFilp, RTProcSelf(), current->pid, current->comm));
726
727 /*
728 * Call common code to create the user session. Associate it with
729 * the file so we can access it in the other methods.
730 */
731 rc = VBoxGuestCreateUserSession(&g_DevExt, &pSession);
732 if (RT_SUCCESS(rc))
733 pFilp->private_data = pSession;
734
735 Log(("vboxguestLinuxOpen: g_DevExt=%p pSession=%p rc=%d/%d (pid=%d/%d %s)\n",
736 &g_DevExt, pSession, rc, vboxguestLinuxConvertToNegErrno(rc),
737 RTProcSelf(), current->pid, current->comm));
738 return vboxguestLinuxConvertToNegErrno(rc);
739}
740
741
742/**
743 * Close device.
744 *
745 * @param pInode Pointer to inode info structure.
746 * @param pFilp Associated file pointer.
747 */
748static int vboxguestLinuxRelease(struct inode *pInode, struct file *pFilp)
749{
750 Log(("vboxguestLinuxRelease: pFilp=%p pSession=%p pid=%d/%d %s\n",
751 pFilp, pFilp->private_data, RTProcSelf(), current->pid, current->comm));
752
753 VBoxGuestCloseSession(&g_DevExt, (PVBOXGUESTSESSION)pFilp->private_data);
754 pFilp->private_data = NULL;
755 return 0;
756}
757
758
759/**
760 * Device I/O Control entry point.
761 *
762 * @param pFilp Associated file pointer.
763 * @param uCmd The function specified to ioctl().
764 * @param ulArg The argument specified to ioctl().
765 */
766#ifdef HAVE_UNLOCKED_IOCTL
767static long vboxguestLinuxIOCtl(struct file *pFilp, unsigned int uCmd, unsigned long ulArg)
768#else
769static int vboxguestLinuxIOCtl(struct inode *pInode, struct file *pFilp, unsigned int uCmd, unsigned long ulArg)
770#endif
771{
772 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pFilp->private_data;
773 uint32_t cbData = _IOC_SIZE(uCmd);
774 void *pvBufFree;
775 void *pvBuf;
776 int rc;
777 uint64_t au64Buf[32/sizeof(uint64_t)];
778
779 Log6(("vboxguestLinuxIOCtl: pFilp=%p uCmd=%#x ulArg=%p pid=%d/%d\n", pFilp, uCmd, (void *)ulArg, RTProcSelf(), current->pid));
780
781 /*
782 * Buffer the request.
783 */
784 if (cbData <= sizeof(au64Buf))
785 {
786 pvBufFree = NULL;
787 pvBuf = &au64Buf[0];
788 }
789 else
790 {
791 pvBufFree = pvBuf = RTMemTmpAlloc(cbData);
792 if (RT_UNLIKELY(!pvBuf))
793 {
794 LogRel((DEVICE_NAME "::IOCtl: RTMemTmpAlloc failed to alloc %u bytes.\n", cbData));
795 return -ENOMEM;
796 }
797 }
798 if (RT_LIKELY(copy_from_user(pvBuf, (void *)ulArg, cbData) == 0))
799 {
800 /*
801 * Process the IOCtl.
802 */
803 size_t cbDataReturned;
804 rc = VBoxGuestCommonIOCtl(uCmd, &g_DevExt, pSession, pvBuf, cbData, &cbDataReturned);
805
806 /*
807 * Copy ioctl data and output buffer back to user space.
808 */
809 if (RT_SUCCESS(rc))
810 {
811 rc = 0;
812 if (RT_UNLIKELY(cbDataReturned > cbData))
813 {
814 LogRel((DEVICE_NAME "::IOCtl: too much output data %u expected %u\n", cbDataReturned, cbData));
815 cbDataReturned = cbData;
816 }
817 if (cbDataReturned > 0)
818 {
819 if (RT_UNLIKELY(copy_to_user((void *)ulArg, pvBuf, cbDataReturned) != 0))
820 {
821 LogRel((DEVICE_NAME "::IOCtl: copy_to_user failed; pvBuf=%p ulArg=%p cbDataReturned=%u uCmd=%d\n",
822 pvBuf, (void *)ulArg, cbDataReturned, uCmd, rc));
823 rc = -EFAULT;
824 }
825 }
826 }
827 else
828 {
829 Log(("vboxguestLinuxIOCtl: pFilp=%p uCmd=%#x ulArg=%p failed, rc=%d\n", pFilp, uCmd, (void *)ulArg, rc));
830 rc = -rc; Assert(rc > 0); /* Positive returns == negated VBox error status codes. */
831 }
832 }
833 else
834 {
835 Log((DEVICE_NAME "::IOCtl: copy_from_user(,%#lx, %#x) failed; uCmd=%#x.\n", ulArg, cbData, uCmd));
836 rc = -EFAULT;
837 }
838 if (pvBufFree)
839 RTMemFree(pvBufFree);
840
841 Log6(("vboxguestLinuxIOCtl: returns %d (pid=%d/%d)\n", rc, RTProcSelf(), current->pid));
842 return rc;
843}
844
845
846/**
847 * Asynchronous notification activation method.
848 *
849 * @returns 0 on success, negative errno on failure.
850 *
851 * @param fd The file descriptor.
852 * @param pFile The file structure.
853 * @param fOn On/off indicator.
854 */
855static int vboxguestFAsync(int fd, struct file *pFile, int fOn)
856{
857 return fasync_helper(fd, pFile, fOn, &g_pFAsyncQueue);
858}
859
860
861/**
862 * Poll function.
863 *
864 * This returns ready to read if the mouse pointer mode or the pointer position
865 * has changed since last call to read.
866 *
867 * @returns 0 if no changes, POLLIN | POLLRDNORM if there are unseen changes.
868 *
869 * @param pFile The file structure.
870 * @param pPt The poll table.
871 *
872 * @remarks This is probably not really used, X11 is said to use the fasync
873 * interface instead.
874 */
875static unsigned int vboxguestPoll(struct file *pFile, poll_table *pPt)
876{
877 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pFile->private_data;
878 uint32_t u32CurSeq = ASMAtomicUoReadU32(&g_DevExt.u32MousePosChangedSeq);
879 unsigned int fMask = pSession->u32MousePosChangedSeq != u32CurSeq
880 ? POLLIN | POLLRDNORM
881 : 0;
882 poll_wait(pFile, &g_PollEventQueue, pPt);
883 return fMask;
884}
885
886
887/**
888 * Read to go with our poll/fasync response.
889 *
890 * @returns 1 or -EINVAL.
891 *
892 * @param pFile The file structure.
893 * @param pbBuf The buffer to read into.
894 * @param cbRead The max number of bytes to read.
895 * @param poff The current file position.
896 *
897 * @remarks This is probably not really used as X11 lets the driver do its own
898 * event reading. The poll condition is therefore also cleared when we
899 * see VMMDevReq_GetMouseStatus in VBoxGuestCommonIOCtl_VMMRequest.
900 */
901static ssize_t vboxguestRead(struct file *pFile, char *pbBuf, size_t cbRead, loff_t *poff)
902{
903 PVBOXGUESTSESSION pSession = (PVBOXGUESTSESSION)pFile->private_data;
904 uint32_t u32CurSeq = ASMAtomicUoReadU32(&g_DevExt.u32MousePosChangedSeq);
905
906 if (*poff != 0)
907 return -EINVAL;
908
909 /*
910 * Fake a single byte read if we're not up to date with the current mouse position.
911 */
912 if ( pSession->u32MousePosChangedSeq != u32CurSeq
913 && cbRead > 0)
914 {
915 pSession->u32MousePosChangedSeq = u32CurSeq;
916 pbBuf[0] = 0;
917 return 1;
918 }
919 return 0;
920}
921
922
923void VBoxGuestNativeISRMousePollEvent(PVBOXGUESTDEVEXT pDevExt)
924{
925#ifdef VBOXGUEST_WITH_INPUT_DRIVER
926 int rc;
927#endif
928 NOREF(pDevExt);
929
930 /*
931 * Wake up everyone that's in a poll() and post anyone that has
932 * subscribed to async notifications.
933 */
934 Log(("VBoxGuestNativeISRMousePollEvent: wake_up_all\n"));
935 wake_up_all(&g_PollEventQueue);
936 Log(("VBoxGuestNativeISRMousePollEvent: kill_fasync\n"));
937 kill_fasync(&g_pFAsyncQueue, SIGIO, POLL_IN);
938#ifdef VBOXGUEST_WITH_INPUT_DRIVER
939 /* Report events to the kernel input device */
940 g_pMouseStatusReq->mouseFeatures = 0;
941 g_pMouseStatusReq->pointerXPos = 0;
942 g_pMouseStatusReq->pointerYPos = 0;
943 rc = VbglGRPerform(&g_pMouseStatusReq->header);
944 if (RT_SUCCESS(rc))
945 {
946 input_report_abs(g_pInputDevice, ABS_X,
947 g_pMouseStatusReq->pointerXPos);
948 input_report_abs(g_pInputDevice, ABS_Y,
949 g_pMouseStatusReq->pointerYPos);
950# ifdef EV_SYN
951 input_sync(g_pInputDevice);
952# endif
953 }
954#endif
955 Log(("VBoxGuestNativeISRMousePollEvent: done\n"));
956}
957
958
959/* Common code that depend on g_DevExt. */
960#include "VBoxGuestIDC-unix.c.h"
961
962EXPORT_SYMBOL(VBoxGuestIDCOpen);
963EXPORT_SYMBOL(VBoxGuestIDCClose);
964EXPORT_SYMBOL(VBoxGuestIDCCall);
965
966
967#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 0)
968
969/** log and dbg_log parameter setter. */
970static int vboxguestLinuxParamLogGrpSet(const char *pszValue, struct kernel_param *pParam)
971{
972 if (g_fLoggerCreated)
973 {
974 PRTLOGGER pLogger = pParam->name[0] == 'd' ? RTLogDefaultInstance() : RTLogRelDefaultInstance();
975 if (pLogger)
976 RTLogGroupSettings(pLogger, pszValue);
977 }
978 else if (pParam->name[0] != 'd')
979 strlcpy(&g_szLogGrp[0], pszValue, sizeof(g_szLogGrp));
980
981 return 0;
982}
983
984
985/** log and dbg_log parameter getter. */
986static int vboxguestLinuxParamLogGrpGet(char *pszBuf, struct kernel_param *pParam)
987{
988 PRTLOGGER pLogger = pParam->name[0] == 'd' ? RTLogDefaultInstance() : RTLogRelDefaultInstance();
989 *pszBuf = '\0';
990 if (pLogger)
991 RTLogGetGroupSettings(pLogger, pszBuf, _4K);
992 return strlen(pszBuf);
993}
994
995
996/** log and dbg_log_flags parameter setter. */
997static int vboxguestLinuxParamLogFlagsSet(const char *pszValue, struct kernel_param *pParam)
998{
999 if (g_fLoggerCreated)
1000 {
1001 PRTLOGGER pLogger = pParam->name[0] == 'd' ? RTLogDefaultInstance() : RTLogRelDefaultInstance();
1002 if (pLogger)
1003 RTLogFlags(pLogger, pszValue);
1004 }
1005 else if (pParam->name[0] != 'd')
1006 strlcpy(&g_szLogFlags[0], pszValue, sizeof(g_szLogFlags));
1007 return 0;
1008}
1009
1010
1011/** log and dbg_log_flags parameter getter. */
1012static int vboxguestLinuxParamLogFlagsGet(char *pszBuf, struct kernel_param *pParam)
1013{
1014 PRTLOGGER pLogger = pParam->name[0] == 'd' ? RTLogDefaultInstance() : RTLogRelDefaultInstance();
1015 *pszBuf = '\0';
1016 if (pLogger)
1017 RTLogGetFlags(pLogger, pszBuf, _4K);
1018 return strlen(pszBuf);
1019}
1020
1021
1022/** log and dbg_log_dest parameter setter. */
1023static int vboxguestLinuxParamLogDstSet(const char *pszValue, struct kernel_param *pParam)
1024{
1025 if (g_fLoggerCreated)
1026 {
1027 PRTLOGGER pLogger = pParam->name[0] == 'd' ? RTLogDefaultInstance() : RTLogRelDefaultInstance();
1028 if (pLogger)
1029 RTLogDestinations(pLogger, pszValue);
1030 }
1031 else if (pParam->name[0] != 'd')
1032 strlcpy(&g_szLogDst[0], pszValue, sizeof(g_szLogDst));
1033 return 0;
1034}
1035
1036
1037/** log and dbg_log_dest parameter getter. */
1038static int vboxguestLinuxParamLogDstGet(char *pszBuf, struct kernel_param *pParam)
1039{
1040 PRTLOGGER pLogger = pParam->name[0] == 'd' ? RTLogDefaultInstance() : RTLogRelDefaultInstance();
1041 *pszBuf = '\0';
1042 if (pLogger)
1043 RTLogGetDestinations(pLogger, pszBuf, _4K);
1044 return strlen(pszBuf);
1045}
1046
1047/*
1048 * Define module parameters.
1049 */
1050module_param_call(log, vboxguestLinuxParamLogGrpSet, vboxguestLinuxParamLogGrpGet, NULL, 0664);
1051module_param_call(log_flags, vboxguestLinuxParamLogFlagsSet, vboxguestLinuxParamLogFlagsGet, NULL, 0664);
1052module_param_call(log_dest, vboxguestLinuxParamLogDstSet, vboxguestLinuxParamLogDstGet, NULL, 0664);
1053# ifdef LOG_ENABLED
1054module_param_call(dbg_log, vboxguestLinuxParamLogGrpSet, vboxguestLinuxParamLogGrpGet, NULL, 0664);
1055module_param_call(dbg_log_flags, vboxguestLinuxParamLogFlagsSet, vboxguestLinuxParamLogFlagsGet, NULL, 0664);
1056module_param_call(dbg_log_dest, vboxguestLinuxParamLogDstSet, vboxguestLinuxParamLogDstGet, NULL, 0664);
1057# endif
1058
1059#endif /* 2.6.0 and later */
1060
1061
1062module_init(vboxguestLinuxModInit);
1063module_exit(vboxguestLinuxModExit);
1064
1065MODULE_AUTHOR(VBOX_VENDOR);
1066MODULE_DESCRIPTION(VBOX_PRODUCT " Guest Additions for Linux Module");
1067MODULE_LICENSE("GPL");
1068#ifdef MODULE_VERSION
1069MODULE_VERSION(VBOX_VERSION_STRING);
1070#endif
1071
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