VirtualBox

source: vbox/trunk/src/VBox/Additions/linux/module/vboxmod.c@ 20935

Last change on this file since 20935 was 20935, checked in by vboxsync, 16 years ago

Additions/Linux/vboxadd: missing logging ioctl in release builds

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 57.8 KB
Line 
1/** @file
2 * vboxadd -- VirtualBox Guest Additions for Linux
3 */
4
5/*
6 * Copyright (C) 2006-2007 Sun Microsystems, Inc.
7 *
8 * This file is part of VirtualBox Open Source Edition (OSE), as
9 * available from http://www.virtualbox.org. This file is free software;
10 * you can redistribute it and/or modify it under the terms of the GNU
11 * General Public License (GPL) as published by the Free Software
12 * Foundation, in version 2 as it comes in the "COPYING" file of the
13 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
14 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
15 *
16 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
17 * Clara, CA 95054 USA or visit http://www.sun.com if you need
18 * additional information or have any questions.
19 */
20
21#include "the-linux-kernel.h"
22#include "version-generated.h"
23
24/* #define IRQ_DEBUG */
25/* #define IOCTL_DEBUG */
26#ifdef IOCTL_DEBUG
27# define IOCTL_ENTRY(name, arg) \
28do { \
29 Log(("IOCTL_ENTRY: %s, 0x%x\n", (name), (arg))); \
30} while(0)
31# define IOCTL_EXIT(name, arg) \
32do { \
33 Log(("IOCTL_EXIT: %s, 0x%x\n", (name), (arg))); \
34} while(0)
35#else
36# define IOCTL_ENTRY(name, arg) do { } while(0)
37# define IOCTL_EXIT(name, arg) do { } while(0)
38#endif
39#ifdef IOCTL_LOG_DEBUG
40# define IOCTL_LOG_ENTRY(arg) \
41do { \
42 Log(("IOCTL_ENTRY: Log, 0x%x\n", (arg))); \
43} while(0)
44# define IOCTL_LOG_EXIT(arg) \
45do { \
46 Log(("IOCTL_EXIT: Log, 0x%x\n", (arg))); \
47} while(0)
48#else
49# define IOCTL_LOG_ENTRY(arg) do { } while(0)
50# define IOCTL_LOG_EXIT(arg) do { } while(0)
51#endif
52#ifdef IOCTL_VMM_DEBUG
53# define IOCTL_VMM_ENTRY(arg) \
54do { \
55 Log(("IOCTL_ENTRY: VMMDevReq, 0x%x\n", (arg))); \
56} while(0)
57# define IOCTL_VMM_EXIT(arg) \
58do { \
59 Log(("IOCTL_EXIT: VMMDevReq, 0x%x\n", (arg))); \
60} while(0)
61#else
62# define IOCTL_VMM_ENTRY(arg) do { } while(0)
63# define IOCTL_VMM_EXIT(arg) do { } while(0)
64#endif
65
66#include "vboxmod.h"
67#include "waitcompat.h"
68
69#include <VBox/log.h>
70#include <VBox/VBoxDev.h>
71#include <iprt/asm.h>
72#include <iprt/assert.h>
73#include <iprt/mem.h>
74#include <iprt/memobj.h>
75#include <linux/miscdevice.h>
76#include <linux/poll.h>
77
78#define xstr(s) str(s)
79#define str(s) #s
80
81MODULE_DESCRIPTION("VirtualBox Guest Additions for Linux Module");
82MODULE_AUTHOR("Sun Microsystems, Inc.");
83MODULE_LICENSE("GPL");
84#ifdef MODULE_VERSION
85MODULE_VERSION(VBOX_VERSION_STRING " (interface " xstr(VMMDEV_VERSION) ")");
86#endif
87
88/** device extension structure (we only support one device instance) */
89static VBoxDevice *vboxDev = NULL;
90/** our file node major id (set dynamically) */
91#ifdef CONFIG_VBOXADD_MAJOR
92static unsigned int vbox_major = CONFIG_VBOXADD_MAJOR;
93#else
94static unsigned int vbox_major = 0;
95#endif
96
97DECLVBGL (void *) vboxadd_cmc_open (void)
98{
99 return vboxDev;
100}
101
102DECLVBGL (void) vboxadd_cmc_close (void *opaque)
103{
104 (void) opaque;
105}
106
107EXPORT_SYMBOL (vboxadd_cmc_open);
108EXPORT_SYMBOL (vboxadd_cmc_close);
109
110
111#define MAX_HGCM_CONNECTIONS 1024
112
113/**
114 * Structure for keeping track of HGCM connections owned by user space processes, so that
115 * we can close the connection if a process does not clean up properly (for example if it
116 * was terminated too abruptly).
117 */
118/* We just define a fixed number of these so far. This can be changed if it ever becomes
119 a problem. */
120static struct
121{
122 /** Open file structure that this connection handle is associated with */
123 struct file *filp;
124 /** HGCM connection ID */
125 uint32_t client_id;
126} hgcm_connections[MAX_HGCM_CONNECTIONS]
127=
128{
129 { 0 }
130};
131
132
133/**
134 * This function converts a VBox result code into a Linux error number.
135 * Note that we return 0 (success) for all informational values, as Linux
136 * has no such concept.
137 */
138static int vboxadd_convert_result(int vbox_err)
139{
140 if ( vbox_err > -1000
141 && vbox_err < 1000)
142 return RTErrConvertToErrno(vbox_err);
143 switch (vbox_err)
144 {
145 case VERR_HGCM_SERVICE_NOT_FOUND: return ESRCH;
146 case VINF_HGCM_CLIENT_REJECTED: return 0;
147 case VERR_HGCM_INVALID_CMD_ADDRESS: return EFAULT;
148 case VINF_HGCM_ASYNC_EXECUTE: return 0;
149 case VERR_HGCM_INTERNAL: return EPROTO;
150 case VERR_HGCM_INVALID_CLIENT_ID: return EINVAL;
151 case VINF_HGCM_SAVE_STATE: return 0;
152 /* No reason to return this to a guest */
153 // case VERR_HGCM_SERVICE_EXISTS: return EEXIST;
154 }
155 AssertMsgFailedReturn(("Unhandled error code %Rrc\n", vbox_err), EPROTO);
156}
157
158/**
159 * Register an HGCM connection as being connected with a given file descriptor, so that it
160 * will be closed automatically when that file descriptor is.
161 *
162 * @returns 0 on success or Linux kernel error number
163 * @param clientID the client ID of the HGCM connection
164 * @param filep the file structure that the connection is to be associated with
165 */
166static int vboxadd_register_hgcm_connection(uint32_t client_id, struct file *filp)
167{
168 int i;
169 bool found = false;
170
171 for (i = 0; i < MAX_HGCM_CONNECTIONS; ++i)
172 {
173 Assert(hgcm_connections[i].client_id != client_id);
174 }
175 for (i = 0; (i < MAX_HGCM_CONNECTIONS) && (false == found); ++i)
176 {
177 if (ASMAtomicCmpXchgU32(&hgcm_connections[i].client_id, client_id, 0))
178 {
179 hgcm_connections[i].filp = filp;
180 found = true;
181 }
182 }
183 return found ? 0 : -ENFILE; /* Any ideas for a better error code? */
184}
185
186/**
187 * Unregister an HGCM connection associated with a given file descriptor without closing
188 * the connection.
189 *
190 * @returns 0 on success or Linux kernel error number
191 * @param clientID the client ID of the HGCM connection
192 */
193static int vboxadd_unregister_hgcm_connection_no_close(uint32_t client_id)
194{
195 int i;
196 bool found = false;
197
198 for (i = 0; (i < MAX_HGCM_CONNECTIONS) && (false == found); ++i)
199 {
200 if (hgcm_connections[i].client_id == client_id)
201 {
202 hgcm_connections[i].filp = NULL;
203 hgcm_connections[i].client_id = 0;
204 found = true;
205 }
206 }
207 for (i = 0; i < MAX_HGCM_CONNECTIONS; ++i)
208 {
209 Assert(hgcm_connections[i].client_id != client_id);
210 }
211 return found ? 0 : -ENOENT;
212}
213
214/**
215 * Unregister all HGCM connections associated with a given file descriptor, closing
216 * the connections in the process. This should be called when a file descriptor is
217 * closed.
218 *
219 * @returns 0 on success or Linux kernel error number
220 * @param clientID the client ID of the HGCM connection
221 */
222static int vboxadd_unregister_all_hgcm_connections(struct file *filp)
223{
224 int i;
225
226 for (i = 0; i < MAX_HGCM_CONNECTIONS; ++i)
227 {
228 if (hgcm_connections[i].filp == filp)
229 {
230 VBoxGuestHGCMDisconnectInfo infoDisconnect;
231 infoDisconnect.u32ClientID = hgcm_connections[i].client_id;
232 vboxadd_cmc_call(vboxDev, VBOXGUEST_IOCTL_HGCM_DISCONNECT,
233 &infoDisconnect);
234 hgcm_connections[i].filp = NULL;
235 hgcm_connections[i].client_id = 0;
236 }
237 }
238 return 0;
239}
240
241/**
242 * File open handler
243 *
244 */
245static int vboxadd_open(struct inode *inode, struct file *filp)
246{
247 /* no checks required */
248 return 0;
249}
250
251static void
252vboxadd_wait_for_event (VBoxGuestWaitEventInfo *info)
253{
254 long timeleft;
255 uint32_t cInterruptions = vboxDev->u32GuestInterruptions;
256 uint32_t in_mask = info->u32EventMaskIn;
257
258 info->u32Result = VBOXGUEST_WAITEVENT_OK;
259 if (RT_INDEFINITE_WAIT != info->u32TimeoutIn)
260 {
261 timeleft = wait_event_interruptible_timeout
262 (vboxDev->eventq,
263 (vboxDev->u32Events & in_mask)
264 || (vboxDev->u32GuestInterruptions != cInterruptions),
265 msecs_to_jiffies (info->u32TimeoutIn)
266 );
267 if (vboxDev->u32GuestInterruptions != cInterruptions)
268 {
269 info->u32Result = VBOXGUEST_WAITEVENT_INTERRUPTED;
270 }
271 if (timeleft < 0)
272 {
273 info->u32Result = VBOXGUEST_WAITEVENT_INTERRUPTED;
274 }
275 if (timeleft == 0)
276 {
277 info->u32Result = VBOXGUEST_WAITEVENT_TIMEOUT;
278 }
279 }
280 else
281 {
282 if (wait_event_interruptible(vboxDev->eventq,
283 (vboxDev->u32Events & in_mask)
284 || (vboxDev->u32GuestInterruptions != cInterruptions)
285 )
286 )
287 {
288 info->u32Result = VBOXGUEST_WAITEVENT_INTERRUPTED;
289 }
290 }
291 info->u32EventFlagsOut = vboxDev->u32Events & in_mask;
292 vboxDev->u32Events &= ~in_mask;
293}
294
295/**
296 * IOCtl handler - wait for an event from the host.
297 *
298 * @returns Linux kernel return code
299 * @param ptr User space pointer to a structure describing the event
300 */
301static int vboxadd_wait_event(void *ptr)
302{
303 int rc = 0;
304 VBoxGuestWaitEventInfo info;
305
306 if (copy_from_user (&info, ptr, sizeof (info)))
307 {
308 LogRelFunc (("VBOXGUEST_IOCTL_WAITEVENT: can not get event info\n"));
309 rc = -EFAULT;
310 }
311
312 if (0 == rc)
313 {
314 vboxadd_wait_for_event (&info);
315
316 if (copy_to_user (ptr, &info, sizeof (info)))
317 {
318 LogRelFunc (("VBOXGUEST_IOCTL_WAITEVENT: can not put out_mask\n"));
319 rc = -EFAULT;
320 }
321 }
322 return 0;
323}
324
325/**
326 * IOCTL handler. Initiate an HGCM connection for a user space application. If the connection
327 * succeeds, it will be associated with the file structure used to open it, so that it will be
328 * automatically shut down again if the file descriptor is closed.
329 *
330 * @returns 0 on success, or a Linux kernel errno value
331 * @param filp the file structure with which the application opened the driver
332 * @param userspace_info userspace pointer to the hgcm connection information
333 * (VBoxGuestHGCMConnectInfo structure)
334 * @retval userspace_info userspace pointer to the hgcm connection information
335 */
336static int vboxadd_hgcm_connect(struct file *filp, unsigned long userspace_info)
337{
338 VBoxGuestHGCMConnectInfo info;
339 int rc = 0;
340
341 if (copy_from_user ((void *)&info, (void *)userspace_info,
342 sizeof (info)) != 0)
343 {
344 LogFunc (("VBOXGUEST_IOCTL_HGCM_CONNECT: can not get connection info\n"));
345 rc = -EFAULT;
346 }
347 info.u32ClientID = 0;
348 if (rc >= 0)
349 {
350 int vrc = vboxadd_cmc_call(vboxDev, VBOXGUEST_IOCTL_HGCM_CONNECT,
351 &info);
352 rc = RT_FAILURE(vrc) ? -vboxadd_convert_result(vrc)
353 : -vboxadd_convert_result(info.result);
354 if (rc < 0)
355 LogFunc(("hgcm connection failed. internal ioctl result %Rrc, hgcm result %Rrc\n",
356 vrc, info.result));
357 if (rc >= 0 && info.result < 0)
358 rc = info.result;
359 }
360 if (rc >= 0)
361 {
362 /* Register that the connection is associated with this file pointer. */
363 LogFunc(("Connected, client ID %u\n", info.u32ClientID));
364 rc = vboxadd_register_hgcm_connection(info.u32ClientID, filp);
365 if (rc < 0)
366 LogFunc(("failed to register the HGCM connection\n"));
367 }
368 if ( rc >= 0
369 && copy_to_user ((void *)userspace_info, (void *)&info,
370 sizeof(info)) != 0)
371 {
372 LogFunc (("failed to return the connection structure\n"));
373 rc = -EFAULT;
374 }
375 if (rc < 0 && (info.u32ClientID != 0))
376 /* Unregister again, as we didn't get as far as informing userspace. */
377 vboxadd_unregister_hgcm_connection_no_close(info.u32ClientID);
378 if (rc < 0 && info.u32ClientID != 0)
379 {
380 /* Disconnect the hgcm connection again, as we told userspace it failed. */
381 VBoxGuestHGCMDisconnectInfo infoDisconnect;
382 infoDisconnect.u32ClientID = info.u32ClientID;
383 vboxadd_cmc_call(vboxDev, VBOXGUEST_IOCTL_HGCM_DISCONNECT,
384 &infoDisconnect);
385 }
386 return rc;
387}
388
389/**
390 * IOCTL handler. Disconnect a specific HGCM connection.
391 *
392 * @returns 0 on success, or a Linux kernel errno value
393 * @param filp the file structure with which the application opened the driver
394 * @param userspace_info userspace pointer to the hgcm connection information
395 * (VBoxGuestHGCMConnectInfo structure)
396 * @retval userspace_info userspace pointer to the hgcm connection information
397 */
398static int vboxadd_hgcm_disconnect(struct file *filp, unsigned long userspace_info)
399{
400 int rc = 0, vrc = VINF_SUCCESS;
401
402 VBoxGuestHGCMDisconnectInfo info;
403 if (copy_from_user((void *)&info, (void *)userspace_info, sizeof (info)) != 0)
404 {
405 LogRelFunc (("VBOXGUEST_IOCTL_HGCM_DISCONNECT: can not get info\n"));
406 rc = -EFAULT;
407 }
408 if (rc >= 0)
409 {
410 vrc = vboxadd_cmc_call(vboxDev, VBOXGUEST_IOCTL_HGCM_DISCONNECT, &info);
411 rc = -vboxadd_convert_result(vrc);
412 if (rc < 0)
413 LogFunc(("HGCM disconnect failed, error %Rrc\n", vrc));
414 }
415 if ( rc >= 0
416 && copy_to_user((void *)userspace_info, (void *)&info, sizeof(info)) != 0)
417 {
418 LogRelFunc (("VBOXGUEST_IOCTL_HGCM_DISCONNECT: failed to return the connection structure\n"));
419 rc = -EFAULT;
420 }
421 return rc;
422}
423
424/** Bounce buffer structure for hcgm guest-host data copies. */
425typedef struct hgcm_bounce_buffer
426{
427 /** Kernel memory address. */
428 void *pKernel;
429 /** User memory address. */
430 void *pUser;
431 /** Buffer size. */
432 size_t cb;
433} hgcm_bounce_buffer;
434
435/** Create a bounce buffer in kernel space for user space memory. */
436static int vboxadd_hgcm_alloc_buffer(hgcm_bounce_buffer **ppBuf, void *pUser,
437 size_t cb, bool copy)
438{
439 hgcm_bounce_buffer *pBuf = NULL;
440 void *pKernel = NULL;
441 int rc = 0;
442
443 AssertPtrReturn(ppBuf, -EINVAL);
444 /* Empty buffers are allowed, but then the user pointer should be NULL. */
445 AssertReturn((cb > 0 && VALID_PTR(pUser)) || (cb == 0 && pUser == NULL),
446 -EINVAL);
447
448 pBuf = RTMemAlloc(sizeof(*pBuf));
449 if (pBuf == NULL)
450 rc = -ENOMEM;
451 if (rc >= 0)
452 {
453 if (cb > 0)
454 {
455 pKernel = RTMemAlloc(cb);
456 if (pKernel == NULL)
457 rc = -ENOMEM;
458 if ( rc >= 0
459 && copy
460 && copy_from_user(pKernel, pUser, cb) != 0)
461 rc = -EFAULT;
462 }
463 else
464 /* We definitely don't want to copy anything from an empty
465 * buffer. */
466 pKernel = NULL;
467 }
468 if (rc >= 0)
469 {
470 pBuf->pKernel = pKernel;
471 pBuf->pUser = pUser;
472 pBuf->cb = cb;
473 *ppBuf = pBuf;
474 }
475 else
476 {
477 RTMemFree(pBuf);
478 RTMemFree(pKernel);
479 LogFunc(("failed, returning %d\n", rc));
480 }
481 return rc;
482}
483
484/** Free a kernel space bounce buffer for user space memory. */
485static int vboxadd_hgcm_free_buffer(hgcm_bounce_buffer *pBuf, bool copy)
486{
487 int rc = 0;
488 AssertPtrReturn(pBuf, -EINVAL);
489 if ((pBuf->cb > 0)
490 && copy
491 && copy_to_user(pBuf->pUser, pBuf->pKernel, pBuf->cb) != 0)
492 rc = -EFAULT;
493 RTMemFree(pBuf->pKernel); /* We want to do this whatever the outcome. */
494 RTMemFree(pBuf);
495 if (rc < 0)
496 LogFunc(("failed, returning %d\n", rc));
497 return rc;
498}
499
500/** Lock down R3 memory as needed for the HGCM call. Copied from
501 * HGCMInternal.cpp and SysHlp.cpp */
502static int vboxadd_buffer_hgcm_parms(void **ppvCtx, VBoxGuestHGCMCallInfo *pCallInfo)
503{
504 uint32_t cbParms = pCallInfo->cParms * sizeof (HGCMFunctionParameter);
505 int rc = 0;
506 unsigned iParm;
507 HGCMFunctionParameter *pParm;
508 memset (ppvCtx, 0, sizeof(void *) * pCallInfo->cParms);
509 if (cbParms)
510 {
511 /* Lock user buffers. */
512 pParm = VBOXGUEST_HGCM_CALL_PARMS(pCallInfo);
513
514 for (iParm = 0; iParm < pCallInfo->cParms; iParm++, pParm++)
515 {
516 switch (pParm->type)
517 {
518 case VMMDevHGCMParmType_LinAddr_Locked_In:
519 pParm->type = VMMDevHGCMParmType_LinAddr_In;
520 break;
521 case VMMDevHGCMParmType_LinAddr_Locked_Out:
522 pParm->type = VMMDevHGCMParmType_LinAddr_Out;
523 break;
524 case VMMDevHGCMParmType_LinAddr_Locked:
525 pParm->type = VMMDevHGCMParmType_LinAddr;
526 break;
527
528 case VMMDevHGCMParmType_LinAddr_In:
529 case VMMDevHGCMParmType_LinAddr_Out:
530 case VMMDevHGCMParmType_LinAddr:
531 {
532 void *pv = (void *) pParm->u.Pointer.u.linearAddr;
533 uint32_t u32Size = pParm->u.Pointer.size;
534 hgcm_bounce_buffer *MemObj = NULL;
535 rc = vboxadd_hgcm_alloc_buffer(&MemObj, pv, u32Size,
536 pParm->type != VMMDevHGCMParmType_LinAddr_Out /* copy */);
537 if (rc >= 0)
538 {
539 ppvCtx[iParm] = MemObj;
540 pParm->u.Pointer.u.linearAddr = (uintptr_t)MemObj->pKernel;
541 }
542 else
543 ppvCtx[iParm] = NULL;
544 break;
545 }
546 default:
547 /* make gcc happy */
548 break;
549 }
550 if (rc < 0)
551 break;
552 }
553 }
554 return rc;
555}
556
557/** Unlock R3 memory after the HGCM call. Copied from HGCMInternal.cpp and
558 * SysHlp.cpp */
559static int vboxadd_unbuffer_hgcm_parms(void **ppvCtx, VBoxGuestHGCMCallInfo *pCallInfo)
560{
561 int rc = 0;
562 unsigned iParm;
563 /* Unlock user buffers. */
564 HGCMFunctionParameter *pParm = VBOXGUEST_HGCM_CALL_PARMS(pCallInfo);
565
566 for (iParm = 0; iParm < pCallInfo->cParms; iParm++, pParm++)
567 {
568 if ( pParm->type == VMMDevHGCMParmType_LinAddr_In
569 || pParm->type == VMMDevHGCMParmType_LinAddr_Out
570 || pParm->type == VMMDevHGCMParmType_LinAddr)
571 {
572 if (ppvCtx[iParm] != NULL)
573 {
574 hgcm_bounce_buffer *MemObj = (hgcm_bounce_buffer *)ppvCtx[iParm];
575 int rc2 = vboxadd_hgcm_free_buffer(MemObj,
576 pParm->type != VMMDevHGCMParmType_LinAddr_In /* copy */);
577 if (rc >= 0 && rc2 < 0)
578 rc = rc2; /* Report the first error. */
579 }
580 }
581 else
582 {
583 if (ppvCtx[iParm] != NULL)
584 {
585 AssertFailed();
586 rc = -EOVERFLOW; /* Something unlikely to turn up elsewhere so
587 * we can see where it's coming from. */
588 }
589 }
590 }
591 return rc;
592}
593
594/**
595 * IOCTL handler. Make an HGCM call.
596 *
597 * @returns 0 on success, or a Linux kernel errno value
598 * @param userspace_info userspace pointer to the hgcm connection information
599 * (VBoxGuestHGCMConnectInfo structure). This will be
600 * updated on success.
601 * @param u32Size the size of the userspace structure
602 */
603static int vboxadd_hgcm_call(unsigned long userspace_info, uint32_t u32Size)
604{
605 VBoxGuestHGCMCallInfo *pInfo = NULL;
606 void *apvCtx[VBOX_HGCM_MAX_PARMS];
607 unsigned haveParms = 0;
608 int rc = 0;
609
610 pInfo = kmalloc(u32Size, GFP_KERNEL);
611 if (pInfo == NULL)
612 rc = -ENOMEM;
613 if ( rc >= 0
614 && 0 != copy_from_user ((void *)pInfo, (void *)userspace_info, u32Size))
615 {
616 LogRelFunc (("can not get info from user space\n"));
617 rc = -EFAULT;
618 }
619 if ( rc >= 0
620 && sizeof(*pInfo) + pInfo->cParms * sizeof(HGCMFunctionParameter) != u32Size)
621 {
622 LogRelFunc (("bad parameter size, structure says %d, ioctl says %d\n",
623 sizeof(*pInfo) + pInfo->cParms * sizeof(HGCMFunctionParameter),
624 u32Size));
625 rc = -EINVAL;
626 }
627 if (rc >= 0)
628 {
629 haveParms = 1;
630 rc = vboxadd_buffer_hgcm_parms(apvCtx, pInfo);
631 }
632 if (rc >= 0)
633 {
634 int vrc;
635 vrc = vboxadd_cmc_call(vboxDev,
636 VBOXGUEST_IOCTL_HGCM_CALL(u32Size), pInfo);
637 rc = -vboxadd_convert_result(vrc);
638 if (rc < 0)
639 LogFunc(("HGCM call failed, error %Rrc\n", vrc));
640 if ( rc >= 0
641 && copy_to_user ((void *)userspace_info, (void *)pInfo,
642 u32Size))
643 {
644 LogRelFunc (("failed to return the information to user space\n"));
645 rc = -EFAULT;
646 }
647 }
648 if (haveParms)
649 {
650 int rc2 = vboxadd_unbuffer_hgcm_parms(apvCtx, pInfo);
651 if (rc >= 0 && rc2 < 0)
652 rc = rc2;
653 }
654 if (pInfo != NULL)
655 kfree(pInfo);
656 return rc;
657}
658
659/**
660 * IOCTL handler. Make an HGCM call with timeout.
661 *
662 * @returns 0 on success, or a Linux kernel errno value
663 * @param userspace_info userspace pointer to the hgcm connection information
664 * (VBoxGuestHGCMConnectInfo structure). This will be
665 * updated on success.
666 * @param u32Size the size of the userspace structure
667 */
668static int vboxadd_hgcm_call_timed(unsigned long userspace_info,
669 uint32_t u32Size)
670{
671 VBoxGuestHGCMCallInfoTimed *pInfo = NULL;
672 void *apvCtx[VBOX_HGCM_MAX_PARMS];
673 unsigned haveParms = 0;
674 int rc = 0;
675
676 pInfo = kmalloc(u32Size, GFP_KERNEL);
677 if (pInfo == NULL)
678 rc = -ENOMEM;
679 if ( rc >= 0
680 && 0 != copy_from_user ((void *)pInfo, (void *)userspace_info, u32Size))
681 {
682 LogRelFunc (("can not get info from user space\n"));
683 rc = -EFAULT;
684 }
685 if ( rc >= 0
686 && sizeof(*pInfo) + pInfo->info.cParms * sizeof(HGCMFunctionParameter) != u32Size)
687 {
688 LogRelFunc (("bad parameter size, structure says %d, ioctl says %d\n",
689 sizeof(*pInfo) + pInfo->info.cParms * sizeof(HGCMFunctionParameter),
690 u32Size));
691 rc = -EINVAL;
692 }
693 if (rc >= 0)
694 {
695 haveParms = 1;
696 rc = vboxadd_buffer_hgcm_parms(apvCtx, &pInfo->info);
697 }
698 if (rc >= 0)
699 {
700 int vrc;
701 pInfo->fInterruptible = true; /* User space may not do uninterruptible waits */
702 vrc = vboxadd_cmc_call(vboxDev,
703 VBOXGUEST_IOCTL_HGCM_CALL_TIMED(u32Size), pInfo);
704 rc = -vboxadd_convert_result(vrc);
705 if (rc < 0)
706 LogFunc(("HGCM call failed, error %Rrc", vrc));
707 if ( rc >= 0
708 && copy_to_user ((void *)userspace_info, (void *)pInfo, u32Size))
709 {
710 LogRelFunc (("failed to return the information to user space\n"));
711 rc = -EFAULT;
712 }
713 }
714 if (haveParms)
715 {
716 int rc2 = vboxadd_unbuffer_hgcm_parms(apvCtx, &pInfo->info);
717 if (rc >= 0 && rc2 < 0)
718 rc = rc2;
719 }
720 if (pInfo != NULL)
721 kfree(pInfo);
722 return rc;
723}
724
725/**
726 * IOCtl handler. Control the interrupt filter mask to specify which VMMDev interrupts
727 * we know how to handle.
728 *
729 * @returns iprt status code
730 * @param pInfo kernel space pointer to the filter mask change info
731 */
732static int vboxadd_control_filter_mask(VBoxGuestFilterMaskInfo *pInfo)
733{
734 VMMDevCtlGuestFilterMask *pReq = NULL;
735 int rc = VbglGRAlloc((VMMDevRequestHeader **)&pReq, sizeof(*pReq), VMMDevReq_CtlGuestFilterMask);
736
737 LogFlow(("VBoxGuestCommonIOCtl: CTL_FILTER_MASK: request received, u32OrMask=0x%x, u32NotMask=0x%x\n", pInfo->u32OrMask, pInfo->u32NotMask));
738 if (RT_FAILURE(rc))
739 Log(("VBoxGuestCommonIOCtl: CTL_FILTER_MASK: failed to allocate %u (%#x) bytes to cache the request. rc=%d!!\n", sizeof(*pReq), sizeof(*pReq), rc));
740 else
741 {
742 pReq->u32OrMask = pInfo->u32OrMask;
743 pReq->u32NotMask = pInfo->u32NotMask;
744 rc = VbglGRPerform(&pReq->header);
745 }
746 if (RT_FAILURE(rc))
747 Log(("VBoxGuestCommonIOCtl: CTL_FILTER_MASK: VbglGRPerform failed, rc=%Rrc!\n", rc));
748 else if (RT_FAILURE(pReq->header.rc))
749 {
750 Log(("VBoxGuestCommonIOCtl: CTL_FILTER_MASK: The request failed; VMMDev rc=%Rrc!\n", pReq->header.rc));
751 rc = pReq->header.rc;
752 }
753 if (pReq)
754 VbglGRFree(&pReq->header);
755 return rc;
756}
757
758/**
759 * IOCTL handler for vboxadd
760 */
761static int vboxadd_ioctl(struct inode *inode, struct file *filp,
762 unsigned int cmd, unsigned long arg)
763{
764 int rc = 0;
765
766 /* Deal with variable size ioctls first. */
767 if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_LOG(0))
768 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
769 {
770 char *pszMessage;
771
772 IOCTL_LOG_ENTRY(arg);
773 pszMessage = kmalloc(_IOC_SIZE(cmd), GFP_KERNEL);
774 if (NULL == pszMessage)
775 {
776 LogRelFunc(("VBOXGUEST_IOCTL_LOG: cannot allocate %d bytes of memory!\n",
777 _IOC_SIZE(cmd)));
778 rc = -ENOMEM;
779 }
780 if ( (0 == rc)
781 && copy_from_user(pszMessage, (void*)arg, _IOC_SIZE(cmd)))
782 {
783 LogRelFunc(("VBOXGUEST_IOCTL_LOG: copy_from_user failed!\n"));
784 rc = -EFAULT;
785 }
786 if (0 == rc)
787 {
788 Log(("%.*s", _IOC_SIZE(cmd), pszMessage));
789 }
790 if (NULL != pszMessage)
791 {
792 kfree(pszMessage);
793 }
794 IOCTL_LOG_EXIT(arg);
795 }
796 else if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_VMMREQUEST(0))
797 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
798 {
799 VMMDevRequestHeader reqHeader;
800 VMMDevRequestHeader *reqFull = NULL;
801 size_t cbRequestSize;
802 size_t cbVanillaRequestSize;
803
804 IOCTL_VMM_ENTRY(arg);
805 if (copy_from_user(&reqHeader, (void*)arg, sizeof(reqHeader)))
806 {
807 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: copy_from_user failed for vmm request!\n"));
808 rc = -EFAULT;
809 }
810 if (0 == rc)
811 {
812 /* get the request size */
813 cbVanillaRequestSize = vmmdevGetRequestSize(reqHeader.requestType);
814 if (!cbVanillaRequestSize)
815 {
816 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: invalid request type: %d\n",
817 reqHeader.requestType));
818 rc = -EINVAL;
819 }
820 }
821 if (0 == rc)
822 {
823 cbRequestSize = reqHeader.size;
824 if (cbRequestSize < cbVanillaRequestSize)
825 {
826 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: invalid request size: %d min: %d type: %d\n",
827 cbRequestSize,
828 cbVanillaRequestSize,
829 reqHeader.requestType));
830 rc = -EINVAL;
831 }
832 }
833 if (0 == rc)
834 {
835 /* request storage for the full request */
836 rc = VbglGRAlloc(&reqFull, cbRequestSize, reqHeader.requestType);
837 if (RT_FAILURE(rc))
838 {
839 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: could not allocate request structure! rc = %d\n", rc));
840 rc = -EFAULT;
841 }
842 }
843 if (0 == rc)
844 {
845 /* now get the full request */
846 if (copy_from_user(reqFull, (void*)arg, cbRequestSize))
847 {
848 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: failed to fetch full request from user space!\n"));
849 rc = -EFAULT;
850 }
851 }
852
853 /* now issue the request */
854 if (0 == rc)
855 {
856 int rrc = VbglGRPerform(reqFull);
857
858 /* asynchronous processing? */
859 if (rrc == VINF_HGCM_ASYNC_EXECUTE)
860 {
861 VMMDevHGCMRequestHeader *reqHGCM = (VMMDevHGCMRequestHeader*)reqFull;
862 wait_event_interruptible (vboxDev->eventq, reqHGCM->fu32Flags & VBOX_HGCM_REQ_DONE);
863 rrc = reqFull->rc;
864 }
865
866 /* failed? */
867 if (RT_FAILURE(rrc) || RT_FAILURE(reqFull->rc))
868 {
869 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: request execution failed!\n"));
870 rc = RT_FAILURE(rrc) ? -RTErrConvertToErrno(rrc)
871 : -RTErrConvertToErrno(reqFull->rc);
872 }
873 else
874 {
875 /* success, copy the result data to user space */
876 if (copy_to_user((void*)arg, (void*)reqFull, cbRequestSize))
877 {
878 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: error copying request result to user space!\n"));
879 rc = -EFAULT;
880 }
881 }
882 }
883 if (NULL != reqFull)
884 VbglGRFree(reqFull);
885 IOCTL_VMM_EXIT(arg);
886 }
887 else if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_HGCM_CALL(0))
888 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
889 {
890 /* Do the HGCM call using the Vbgl bits */
891 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CALL", arg);
892 rc = vboxadd_hgcm_call(arg, _IOC_SIZE(cmd));
893 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CALL", arg);
894 }
895 else if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_HGCM_CALL_TIMED(0))
896 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
897 {
898 /* Do the HGCM call using the Vbgl bits */
899 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CALL_TIMED", arg);
900 rc = vboxadd_hgcm_call_timed(arg, _IOC_SIZE(cmd));
901 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CALL_TIMED", arg);
902 }
903 else
904 {
905 switch (cmd)
906 {
907 case VBOXGUEST_IOCTL_WAITEVENT:
908 IOCTL_ENTRY("VBOXGUEST_IOCTL_WAITEVENT", arg);
909 rc = vboxadd_wait_event((void *) arg);
910 IOCTL_EXIT("VBOXGUEST_IOCTL_WAITEVENT", arg);
911 break;
912 case VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS:
913 IOCTL_ENTRY("VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS", arg);
914 ++vboxDev->u32GuestInterruptions;
915 IOCTL_EXIT("VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS", arg);
916 break;
917 case VBOXGUEST_IOCTL_HGCM_CONNECT:
918 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CONNECT", arg);
919 rc = vboxadd_hgcm_connect(filp, arg);
920 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CONNECT", arg);
921 break;
922 case VBOXGUEST_IOCTL_HGCM_DISCONNECT:
923 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_DISCONNECT", arg);
924 vboxadd_hgcm_disconnect(filp, arg);
925 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_DISCONNECT", arg);
926 break;
927 case VBOXGUEST_IOCTL_CTL_FILTER_MASK:
928 {
929 VBoxGuestFilterMaskInfo info;
930 IOCTL_ENTRY("VBOXGUEST_IOCTL_CTL_FILTER_MASK", arg);
931 if (copy_from_user((void*)&info, (void*)arg, sizeof(info)))
932 {
933 LogRelFunc(("VBOXGUEST_IOCTL_CTL_FILTER_MASK: error getting parameters from user space!\n"));
934 rc = -EFAULT;
935 break;
936 }
937 rc = -RTErrConvertToErrno(vboxadd_control_filter_mask(&info));
938 IOCTL_EXIT("VBOXGUEST_IOCTL_CTL_FILTER_MASK", arg);
939 break;
940 }
941 default:
942 LogRelFunc(("unknown command: %x\n", cmd));
943 rc = -EINVAL;
944 break;
945 }
946 }
947 return rc;
948}
949
950/**
951 * IOCTL handler for vboxuser
952 * @todo currently this is just a copy of vboxadd_ioctl. We should
953 * decide if we wish to restrict this. If we do, we should remove
954 * the more general ioctls (HGCM call, VMM device request) and
955 * replace them with specific ones. If not, then we should just
956 * make vboxadd world readable and writable or something.
957 */
958static int vboxuser_ioctl(struct inode *inode, struct file *filp,
959 unsigned int cmd, unsigned long arg)
960{
961 int rc = 0;
962
963 /* Deal with variable size ioctls first. */
964 if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_LOG(0))
965 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
966 {
967 char *pszMessage;
968
969 IOCTL_LOG_ENTRY(arg);
970 pszMessage = kmalloc(_IOC_SIZE(cmd), GFP_KERNEL);
971 if (NULL == pszMessage)
972 {
973 LogRelFunc(("VBOXGUEST_IOCTL_LOG: cannot allocate %d bytes of memory!\n",
974 _IOC_SIZE(cmd)));
975 rc = -ENOMEM;
976 }
977 if ( (0 == rc)
978 && copy_from_user(pszMessage, (void*)arg, _IOC_SIZE(cmd)))
979 {
980 LogRelFunc(("VBOXGUEST_IOCTL_LOG: copy_from_user failed!\n"));
981 rc = -EFAULT;
982 }
983 if (0 == rc)
984 /* This only produces output in debug builds */
985 Log(("%.*s", _IOC_SIZE(cmd), pszMessage));
986 if (NULL != pszMessage)
987 {
988 kfree(pszMessage);
989 }
990 IOCTL_LOG_EXIT(arg);
991 }
992 else if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_VMMREQUEST(0))
993 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
994 {
995 VMMDevRequestHeader reqHeader;
996 VMMDevRequestHeader *reqFull = NULL;
997 size_t cbRequestSize;
998 size_t cbVanillaRequestSize;
999
1000 IOCTL_VMM_ENTRY(arg);
1001 if (copy_from_user(&reqHeader, (void*)arg, sizeof(reqHeader)))
1002 {
1003 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: copy_from_user failed for vmm request!\n"));
1004 rc = -EFAULT;
1005 }
1006 if (0 == rc)
1007 {
1008 /* get the request size */
1009 cbVanillaRequestSize = vmmdevGetRequestSize(reqHeader.requestType);
1010 if (!cbVanillaRequestSize)
1011 {
1012 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: invalid request type: %d\n",
1013 reqHeader.requestType));
1014 rc = -EINVAL;
1015 }
1016 }
1017 if (0 == rc)
1018 {
1019 cbRequestSize = reqHeader.size;
1020 if (cbRequestSize < cbVanillaRequestSize)
1021 {
1022 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: invalid request size: %d min: %d type: %d\n",
1023 cbRequestSize,
1024 cbVanillaRequestSize,
1025 reqHeader.requestType));
1026 rc = -EINVAL;
1027 }
1028 }
1029 if (0 == rc)
1030 {
1031 /* request storage for the full request */
1032 rc = VbglGRAlloc(&reqFull, cbRequestSize, reqHeader.requestType);
1033 if (RT_FAILURE(rc))
1034 {
1035 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: could not allocate request structure! rc = %d\n", rc));
1036 rc = -EFAULT;
1037 }
1038 }
1039 if (0 == rc)
1040 {
1041 /* now get the full request */
1042 if (copy_from_user(reqFull, (void*)arg, cbRequestSize))
1043 {
1044 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: failed to fetch full request from user space!\n"));
1045 rc = -EFAULT;
1046 }
1047 }
1048
1049 /* now issue the request */
1050 if (0 == rc)
1051 {
1052 int rrc = VbglGRPerform(reqFull);
1053
1054 /* asynchronous processing? */
1055 if (rrc == VINF_HGCM_ASYNC_EXECUTE)
1056 {
1057 VMMDevHGCMRequestHeader *reqHGCM = (VMMDevHGCMRequestHeader*)reqFull;
1058 wait_event_interruptible (vboxDev->eventq, reqHGCM->fu32Flags & VBOX_HGCM_REQ_DONE);
1059 rrc = reqFull->rc;
1060 }
1061
1062 /* failed? */
1063 if (RT_FAILURE(rrc) || RT_FAILURE(reqFull->rc))
1064 {
1065 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: request execution failed!\n"));
1066 rc = RT_FAILURE(rrc) ? -RTErrConvertToErrno(rrc)
1067 : -RTErrConvertToErrno(reqFull->rc);
1068 }
1069 else
1070 {
1071 /* success, copy the result data to user space */
1072 if (copy_to_user((void*)arg, (void*)reqFull, cbRequestSize))
1073 {
1074 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: error copying request result to user space!\n"));
1075 rc = -EFAULT;
1076 }
1077 }
1078 }
1079 if (NULL != reqFull)
1080 VbglGRFree(reqFull);
1081 IOCTL_VMM_EXIT(arg);
1082 }
1083 else if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_HGCM_CALL(0))
1084 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
1085 {
1086 /* Do the HGCM call using the Vbgl bits */
1087 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CALL", arg);
1088 rc = vboxadd_hgcm_call(arg, _IOC_SIZE(cmd));
1089 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CALL", arg);
1090 }
1091 else if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_HGCM_CALL_TIMED(0))
1092 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
1093 {
1094 /* Do the HGCM call using the Vbgl bits */
1095 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CALL_TIMED", arg);
1096 rc = vboxadd_hgcm_call_timed(arg, _IOC_SIZE(cmd));
1097 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CALL_TIMED", arg);
1098 }
1099 else
1100 {
1101 switch (cmd)
1102 {
1103 case VBOXGUEST_IOCTL_WAITEVENT:
1104 IOCTL_ENTRY("VBOXGUEST_IOCTL_WAITEVENT", arg);
1105 rc = vboxadd_wait_event((void *) arg);
1106 IOCTL_EXIT("VBOXGUEST_IOCTL_WAITEVENT", arg);
1107 break;
1108 case VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS:
1109 IOCTL_ENTRY("VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS", arg);
1110 ++vboxDev->u32GuestInterruptions;
1111 IOCTL_EXIT("VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS", arg);
1112 break;
1113 case VBOXGUEST_IOCTL_HGCM_CONNECT:
1114 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CONNECT", arg);
1115 rc = vboxadd_hgcm_connect(filp, arg);
1116 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CONNECT", arg);
1117 break;
1118 case VBOXGUEST_IOCTL_HGCM_DISCONNECT:
1119 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_DISCONNECT", arg);
1120 vboxadd_hgcm_disconnect(filp, arg);
1121 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_DISCONNECT", arg);
1122 break;
1123 case VBOXGUEST_IOCTL_CTL_FILTER_MASK:
1124 {
1125 VBoxGuestFilterMaskInfo info;
1126 IOCTL_ENTRY("VBOXGUEST_IOCTL_CTL_FILTER_MASK", arg);
1127 if (copy_from_user((void*)&info, (void*)arg, sizeof(info)))
1128 {
1129 LogRelFunc(("VBOXGUEST_IOCTL_CTL_FILTER_MASK: error getting parameters from user space!\n"));
1130 rc = -EFAULT;
1131 break;
1132 }
1133 rc = -RTErrConvertToErrno(vboxadd_control_filter_mask(&info));
1134 IOCTL_EXIT("VBOXGUEST_IOCTL_CTL_FILTER_MASK", arg);
1135 break;
1136 }
1137 default:
1138 LogRelFunc(("unknown command: %x\n", cmd));
1139 rc = -EINVAL;
1140 break;
1141 }
1142 }
1143 return rc;
1144}
1145
1146/**
1147 * Poll function. This returns "ready to read" if the guest is in absolute
1148 * mouse pointer mode and the pointer position has changed since the last
1149 * poll.
1150 */
1151unsigned int
1152vboxadd_poll (struct file *file, poll_table *wait)
1153{
1154 int result = 0;
1155 poll_wait(file, &vboxDev->eventq, wait);
1156 if (vboxDev->u32Events & VMMDEV_EVENT_MOUSE_POSITION_CHANGED)
1157 result = (POLLIN | POLLRDNORM);
1158 vboxDev->u32Events &= ~VMMDEV_EVENT_MOUSE_POSITION_CHANGED;
1159 return result;
1160}
1161
1162/** Asynchronous notification activation method. */
1163static int
1164vboxadd_fasync(int fd, struct file *file, int mode)
1165{
1166 return fasync_helper(fd, file, mode, &vboxDev->async_queue);
1167}
1168
1169/**
1170 * Dummy read function - we only supply this because we implement poll and
1171 * fasync.
1172 */
1173static ssize_t
1174vboxadd_read (struct file *file, char *buf, size_t count, loff_t *loff)
1175{
1176 if (0 == count || *loff != 0)
1177 {
1178 return -EINVAL;
1179 }
1180 buf[0] = 0;
1181 return 1;
1182}
1183
1184/**
1185 * File close handler for vboxadd. Clean up any HGCM connections associated
1186 * with the open file which might still be open.
1187 */
1188static int vboxadd_release(struct inode *inode, struct file * filp)
1189{
1190 vboxadd_unregister_all_hgcm_connections(filp);
1191 /* Deactivate our asynchronous queue. */
1192 vboxadd_fasync(-1, filp, 0);
1193 return 0;
1194}
1195
1196/**
1197 * File close handler for vboxuser. Clean up any HGCM connections associated
1198 * with the open file which might still be open.
1199 */
1200static int vboxuser_release(struct inode *inode, struct file * filp)
1201{
1202 vboxadd_unregister_all_hgcm_connections(filp);
1203 return 0;
1204}
1205
1206/** file operations for the vboxadd device */
1207static struct file_operations vboxadd_fops =
1208{
1209 .owner = THIS_MODULE,
1210 .open = vboxadd_open,
1211 .ioctl = vboxadd_ioctl,
1212 .poll = vboxadd_poll,
1213 .fasync = vboxadd_fasync,
1214 .read = vboxadd_read,
1215 .release = vboxadd_release,
1216 .llseek = no_llseek
1217};
1218
1219/** Miscellaneous device allocation for vboxadd */
1220static struct miscdevice gMiscVBoxAdd =
1221{
1222 minor: MISC_DYNAMIC_MINOR,
1223 name: VBOXADD_NAME,
1224 fops: &vboxadd_fops
1225};
1226
1227/** file operations for the vboxuser device */
1228static struct file_operations vboxuser_fops =
1229{
1230 .owner = THIS_MODULE,
1231 .open = vboxadd_open,
1232 .ioctl = vboxuser_ioctl,
1233 .release = vboxuser_release,
1234 .llseek = no_llseek
1235};
1236
1237/** Miscellaneous device allocation for vboxuser */
1238static struct miscdevice gMiscVBoxUser =
1239{
1240 minor: MISC_DYNAMIC_MINOR,
1241 name: VBOXUSER_NAME,
1242 fops: &vboxuser_fops
1243};
1244
1245#ifndef IRQ_RETVAL
1246/* interrupt handlers in 2.4 kernels don't return anything */
1247# define irqreturn_t void
1248# define IRQ_RETVAL(n)
1249#endif
1250
1251/**
1252 * vboxadd_irq_handler
1253 *
1254 * Interrupt handler
1255 *
1256 * @returns scsi error code
1257 * @param irq Irq number
1258 * @param dev_id Irq handler parameter
1259 * @param regs Regs
1260 *
1261 */
1262#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 19)
1263static irqreturn_t vboxadd_irq_handler(int irq, void *dev_id)
1264#else
1265static irqreturn_t vboxadd_irq_handler(int irq, void *dev_id, struct pt_regs *regs)
1266#endif
1267{
1268 int fIRQTaken = 0;
1269 int rcVBox;
1270
1271#ifdef IRQ_DEBUG
1272 Log(("vboxadd IRQ_DEBUG: vboxDev->pVMMDevMemory=%p vboxDev->pVMMDevMemory->fHaveEvents=%d\n",
1273 vboxDev->pVMMDevMemory, vboxDev->pVMMDevMemory->V.V1_04.fHaveEvents));
1274#endif
1275
1276 /* check if IRQ was asserted by VBox */
1277 if (vboxDev->pVMMDevMemory->V.V1_04.fHaveEvents != 0)
1278 {
1279#ifdef IRQ_DEBUG
1280 Log(("vboxadd IRQ_DEBUG: got IRQ with event mask 0x%x\n",
1281 vboxDev->irqAckRequest->events));
1282#endif
1283
1284 /* make a copy of the event mask */
1285 rcVBox = VbglGRPerform (&vboxDev->irqAckRequest->header);
1286 if (RT_SUCCESS(rcVBox) && RT_SUCCESS(vboxDev->irqAckRequest->header.rc))
1287 {
1288 if (RT_LIKELY (vboxDev->irqAckRequest->events))
1289 {
1290 vboxDev->u32Events |= vboxDev->irqAckRequest->events;
1291 if ( vboxDev->irqAckRequest->events
1292 & VMMDEV_EVENT_MOUSE_POSITION_CHANGED)
1293 kill_fasync(&vboxDev->async_queue, SIGIO, POLL_IN);
1294 wake_up (&vboxDev->eventq);
1295 }
1296 }
1297 else
1298 {
1299 /* impossible... */
1300 LogRelFunc(("IRQ was not acknowledged! rc = %Rrc, header.rc = %Rrc\n",
1301 rcVBox, vboxDev->irqAckRequest->header.rc));
1302 BUG ();
1303 }
1304
1305 /* it was ours! */
1306 fIRQTaken = 1;
1307 }
1308#ifdef IRQ_DEBUG
1309 else
1310 {
1311 /* we might be attached to a shared interrupt together with another device. */
1312 Log(("vboxadd IRQ_DEBUG: stale IRQ mem=%p events=%d devevents=%#x\n",
1313 vboxDev->pVMMDevMemory,
1314 vboxDev->pVMMDevMemory->V.V1_04.fHaveEvents,
1315 vboxDev->u32Events));
1316 }
1317#endif
1318 /* it was ours */
1319 return IRQ_RETVAL(fIRQTaken);
1320}
1321
1322/**
1323 * Helper function to reserve a fixed kernel address space window
1324 * and tell the VMM that it can safely put its hypervisor there.
1325 * This function might fail which is not a critical error.
1326 */
1327static int vboxadd_reserve_hypervisor(void)
1328{
1329 VMMDevReqHypervisorInfo *req = NULL;
1330 int rcVBox;
1331
1332 /* allocate request structure */
1333 rcVBox = VbglGRAlloc(
1334 (VMMDevRequestHeader**)&req,
1335 sizeof(VMMDevReqHypervisorInfo),
1336 VMMDevReq_GetHypervisorInfo
1337 );
1338 if (RT_FAILURE(rcVBox))
1339 {
1340 LogRelFunc(("failed to allocate hypervisor info structure! rc = %Rrc\n", rcVBox));
1341 goto bail_out;
1342 }
1343 /* query the hypervisor information */
1344 rcVBox = VbglGRPerform(&req->header);
1345 if (RT_SUCCESS(rcVBox) && RT_SUCCESS(req->header.rc))
1346 {
1347 /* are we supposed to make a reservation? */
1348 if (req->hypervisorSize)
1349 {
1350 /** @todo repeat this several times until we get an address the host likes */
1351
1352 void *hypervisorArea;
1353 /* reserve another 4MB because the start needs to be 4MB aligned */
1354 uint32_t hypervisorSize = req->hypervisorSize + 0x400000;
1355 /* perform a fictive IO space mapping */
1356 hypervisorArea = ioremap(HYPERVISOR_PHYSICAL_START, hypervisorSize);
1357 if (hypervisorArea)
1358 {
1359 /* communicate result to VMM, align at 4MB */
1360 req->hypervisorStart = (VMMDEVHYPPTR32)(uintptr_t)RT_ALIGN_P(hypervisorArea, 0x400000);
1361 req->header.requestType = VMMDevReq_SetHypervisorInfo;
1362 req->header.rc = VERR_GENERAL_FAILURE;
1363 rcVBox = VbglGRPerform(&req->header);
1364 if (RT_SUCCESS(rcVBox) && RT_SUCCESS(req->header.rc))
1365 {
1366 /* store mapping for future unmapping */
1367 vboxDev->hypervisorStart = hypervisorArea;
1368 vboxDev->hypervisorSize = hypervisorSize;
1369 }
1370 else
1371 {
1372 LogRelFunc(("failed to set hypervisor region! rc = %Rrc, header.rc = %Rrc\n",
1373 rcVBox, req->header.rc));
1374 goto bail_out;
1375 }
1376 }
1377 else
1378 {
1379 LogRelFunc(("failed to allocate 0x%x bytes of IO space\n", hypervisorSize));
1380 goto bail_out;
1381 }
1382 }
1383 }
1384 else
1385 {
1386 LogRelFunc(("failed to query hypervisor info! rc = %Rrc, header.rc = %Rrc\n",
1387 rcVBox, req->header.rc));
1388 goto bail_out;
1389 }
1390 /* successful return */
1391 VbglGRFree(&req->header);
1392 return 0;
1393
1394bail_out:
1395 /* error return */
1396 if (req)
1397 VbglGRFree(&req->header);
1398 return 1;
1399}
1400
1401/**
1402 * Helper function to free the hypervisor address window
1403 *
1404 */
1405static int vboxadd_free_hypervisor(void)
1406{
1407 VMMDevReqHypervisorInfo *req = NULL;
1408 int rcVBox;
1409
1410 /* allocate request structure */
1411 rcVBox = VbglGRAlloc(
1412 (VMMDevRequestHeader**)&req,
1413 sizeof(VMMDevReqHypervisorInfo),
1414 VMMDevReq_SetHypervisorInfo
1415 );
1416 if (RT_FAILURE(rcVBox))
1417 {
1418 LogRelFunc(("failed to allocate hypervisor info structure! rc = %Rrc\n", rcVBox));
1419 goto bail_out;
1420 }
1421 /* reset the hypervisor information */
1422 req->hypervisorStart = 0;
1423 req->hypervisorSize = 0;
1424 rcVBox = VbglGRPerform(&req->header);
1425 if (RT_SUCCESS(rcVBox) && RT_SUCCESS(req->header.rc))
1426 {
1427 /* now we can free the associated IO space mapping */
1428 iounmap(vboxDev->hypervisorStart);
1429 vboxDev->hypervisorStart = 0;
1430 }
1431 else
1432 {
1433 LogRelFunc(("failed to reset hypervisor info! rc = %Rrc, header.rc = %Rrc\n",
1434 rcVBox, req->header.rc));
1435 goto bail_out;
1436 }
1437 return 0;
1438
1439 bail_out:
1440 if (req)
1441 VbglGRFree(&req->header);
1442 return 1;
1443}
1444
1445/**
1446 * Helper to free resources
1447 *
1448 */
1449static void free_resources(void)
1450{
1451 if (vboxDev)
1452 {
1453 {
1454 /* Unregister notifications when the host absolute pointer
1455 * position changes. */
1456 VBoxGuestFilterMaskInfo info;
1457 info.u32OrMask = 0;
1458 info.u32NotMask = VMMDEV_EVENT_MOUSE_POSITION_CHANGED;
1459 vboxadd_control_filter_mask(&info);
1460 }
1461 /* Detach from IRQ before cleaning up! */
1462 if (vboxDev->irq)
1463 free_irq(vboxDev->irq, vboxDev);
1464 if (vboxDev->hypervisorStart)
1465 vboxadd_free_hypervisor();
1466 if (vboxDev->irqAckRequest)
1467 {
1468 VbglGRFree(&vboxDev->irqAckRequest->header);
1469 VbglTerminate();
1470 }
1471 if (vboxDev->pVMMDevMemory)
1472 iounmap(vboxDev->pVMMDevMemory);
1473 if (vboxDev->vmmdevmem)
1474 release_mem_region(vboxDev->vmmdevmem, vboxDev->vmmdevmem_size);
1475 kfree(vboxDev);
1476 vboxDev = NULL;
1477 }
1478}
1479
1480#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 20)
1481#define PCI_DEV_GET(v,d,p) pci_get_device(v,d,p)
1482#define PCI_DEV_PUT(x) pci_dev_put(x)
1483#else
1484#define PCI_DEV_GET(v,d,p) pci_find_device(v,d,p)
1485#define PCI_DEV_PUT(x) do {} while(0)
1486#endif
1487
1488/**
1489 * Module initialization
1490 *
1491 */
1492static __init int init(void)
1493{
1494 int rc = 0, rcVBox = VINF_SUCCESS;
1495 bool fHaveVBoxAdd = false, fHaveVBoxUser = false, fHaveGuestLib = false;
1496 struct pci_dev *pcidev = NULL;
1497
1498 rcVBox = vboxadd_cmc_init();
1499 if (RT_FAILURE(rcVBox))
1500 {
1501 printk (KERN_ERR "vboxadd: could not init cmc, VBox error code %d.\n", rcVBox);
1502 rc = -RTErrConvertToErrno(rcVBox);
1503 }
1504
1505 /* Detect PCI device */
1506 if (!rc)
1507 {
1508 pcidev = PCI_DEV_GET(VMMDEV_VENDORID, VMMDEV_DEVICEID, pcidev);
1509 if (!pcidev)
1510 {
1511 printk(KERN_ERR "vboxadd: VirtualBox Guest PCI device not found.\n");
1512 rc = -ENODEV;
1513 }
1514 }
1515
1516 if (!rc)
1517 {
1518 rc = pci_enable_device (pcidev);
1519 if (rc)
1520 LogRel(("vboxadd: could not enable device: %d\n", rc));
1521 }
1522 if (!rc)
1523 LogRel(("Starting VirtualBox version %s Guest Additions\n",
1524 VBOX_VERSION_STRING));
1525
1526 /* Register vboxadd */
1527 if (!rc && vbox_major > 0) /* Register as a character device in this case */
1528 {
1529 rc = register_chrdev(vbox_major, VBOXADD_NAME, &vboxadd_fops);
1530 if (rc) /* As we pass a non-zero major, rc should be zero on success. */
1531 LogRel(("vboxadd: register_chrdev failed: vbox_major: %d, err = %d\n",
1532 vbox_major, rc));
1533 }
1534 else if (!rc) /* Register as a miscellaneous device otherwise */
1535 {
1536 rc = misc_register(&gMiscVBoxAdd);
1537 if (rc)
1538 LogRel(("vboxadd: misc_register failed for %s (rc=%d)\n",
1539 VBOXADD_NAME, rc));
1540 }
1541 if (!rc)
1542 fHaveVBoxAdd = true;
1543
1544 /* Register our user session device */
1545 if (!rc)
1546 {
1547 rc = misc_register(&gMiscVBoxUser);
1548 if (rc)
1549 LogRel(("vboxadd: misc_register failed for %s (rc=%d)\n",
1550 VBOXUSER_NAME, rc));
1551 }
1552 if (!rc)
1553 fHaveVBoxUser = true;
1554
1555 /* allocate and initialize device extension */
1556 if (!rc)
1557 {
1558 vboxDev = kmalloc(sizeof(*vboxDev), GFP_KERNEL);
1559 if (vboxDev)
1560 memset(vboxDev, 0, sizeof(*vboxDev));
1561 else
1562 {
1563 LogRel(("vboxadd: could not allocate private device structure\n"));
1564 rc = -ENOMEM;
1565 }
1566 }
1567
1568 if (!rc)
1569 {
1570 /* get the IO port region */
1571 vboxDev->io_port = pci_resource_start(pcidev, 0);
1572
1573 /* get the memory region */
1574 vboxDev->vmmdevmem = pci_resource_start(pcidev, 1);
1575 vboxDev->vmmdevmem_size = pci_resource_len(pcidev, 1);
1576
1577 /* all resources found? */
1578 if (!vboxDev->io_port || !vboxDev->vmmdevmem || !vboxDev->vmmdevmem_size)
1579 {
1580 LogRel(("vboxadd: did not find expected hardware resources\n"));
1581 rc = -ENXIO;
1582 }
1583 }
1584
1585 /* request ownership of adapter memory */
1586 if (!rc && !request_mem_region(vboxDev->vmmdevmem, vboxDev->vmmdevmem_size,
1587 VBOXADD_NAME))
1588 {
1589 LogRel(("vboxadd: failed to obtain adapter memory\n"));
1590 rc = -EBUSY;
1591 }
1592
1593 /* map adapter memory into kernel address space and check version */
1594 if (!rc)
1595 {
1596 vboxDev->pVMMDevMemory = (VMMDevMemory *) ioremap(vboxDev->vmmdevmem,
1597 vboxDev->vmmdevmem_size);
1598 if (!vboxDev->pVMMDevMemory)
1599 {
1600 LogRel(("vboxadd: ioremap failed\n"));
1601 rc = -ENOMEM;
1602 }
1603 }
1604
1605 if (!rc && (vboxDev->pVMMDevMemory->u32Version != VMMDEV_MEMORY_VERSION))
1606 {
1607 LogRel(("vboxadd: invalid VMM device memory version! (got 0x%x, expected 0x%x)\n",
1608 vboxDev->pVMMDevMemory->u32Version, VMMDEV_MEMORY_VERSION));
1609 rc = -ENXIO;
1610 }
1611
1612 /* initialize ring 0 guest library */
1613 if (!rc)
1614 {
1615 rcVBox = VbglInit(vboxDev->io_port, vboxDev->pVMMDevMemory);
1616 if (RT_FAILURE(rcVBox))
1617 {
1618 LogRel(("vboxadd: could not initialize VBGL subsystem: %Rrc\n",
1619 rcVBox));
1620 rc = -RTErrConvertToErrno(rcVBox);
1621 }
1622 }
1623 if (!rc)
1624 fHaveGuestLib = true;
1625
1626 /* report guest information to host, this must be done as the very first request */
1627 if (!rc)
1628 {
1629 VMMDevReportGuestInfo *infoReq = NULL;
1630
1631 rcVBox = VbglGRAlloc((VMMDevRequestHeader**)&infoReq,
1632 sizeof(VMMDevReportGuestInfo), VMMDevReq_ReportGuestInfo);
1633 if (RT_FAILURE(rcVBox))
1634 {
1635 LogRel(("vboxadd: could not allocate request structure: %Rrc\n", rcVBox));
1636 rc = -RTErrConvertToErrno(rcVBox);
1637 }
1638 /* report guest version to host, the VMMDev requires that to be done
1639 * before any other VMMDev operations. */
1640 if (infoReq)
1641 {
1642 infoReq->guestInfo.additionsVersion = VMMDEV_VERSION;
1643#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 0)
1644 infoReq->guestInfo.osType = VBOXOSTYPE_Linux26;
1645#else
1646 infoReq->guestInfo.osType = VBOXOSTYPE_Linux24;
1647#endif
1648 rcVBox = VbglGRPerform(&infoReq->header);
1649 }
1650 if ( infoReq
1651 && ( RT_FAILURE(rcVBox)
1652 || RT_FAILURE(infoReq->header.rc)))
1653 {
1654 LogRel(("vboxadd: error reporting guest information to host: %Rrc, header: %Rrc\n",
1655 rcVBox, infoReq->header.rc));
1656 rc = RT_FAILURE(rcVBox) ? -RTErrConvertToErrno(rcVBox)
1657 : -RTErrConvertToErrno(infoReq->header.rc);
1658 }
1659 if (infoReq)
1660 VbglGRFree(&infoReq->header);
1661 }
1662
1663 /* Unset the graphics capability until/unless X is loaded. */
1664 /** @todo check the error code once we bump the additions version.
1665 For now we ignore it for compatibility with older hosts. */
1666 if (!rc)
1667 {
1668 VMMDevReqGuestCapabilities2 *vmmreqGuestCaps;
1669
1670 rcVBox = VbglGRAlloc((VMMDevRequestHeader**)&vmmreqGuestCaps,
1671 sizeof(VMMDevReqGuestCapabilities2),
1672 VMMDevReq_SetGuestCapabilities);
1673 if (RT_FAILURE(rcVBox))
1674 {
1675 LogRel(("vboxadd: could not allocate request structure: %Rrc\n",
1676 rcVBox));
1677 rc = -RTErrConvertToErrno(rcVBox);
1678 }
1679 else
1680 {
1681 vmmreqGuestCaps->u32OrMask = 0;
1682 vmmreqGuestCaps->u32NotMask = VMMDEV_GUEST_SUPPORTS_GRAPHICS;
1683 rcVBox = VbglGRPerform(&vmmreqGuestCaps->header);
1684 if (RT_FAILURE(rcVBox))
1685 {
1686 LogRel(("vboxadd: could not allocate request structure: %Rrc\n",
1687 rcVBox));
1688 rc = -RTErrConvertToErrno(rcVBox);
1689 }
1690 VbglGRFree(&vmmreqGuestCaps->header);
1691 }
1692 }
1693
1694 /* perform hypervisor address space reservation */
1695 if (!rc && vboxadd_reserve_hypervisor())
1696 {
1697 /* we just ignore the error, no address window reservation, non fatal */
1698 }
1699
1700 /* allocate a VMM request structure for use in the ISR */
1701 if (!rc)
1702 {
1703 rcVBox = VbglGRAlloc((VMMDevRequestHeader**)&vboxDev->irqAckRequest,
1704 sizeof(VMMDevEvents), VMMDevReq_AcknowledgeEvents);
1705 if (RT_FAILURE(rcVBox))
1706 {
1707 LogRel(("vboxadd: could not allocate request structure: %Rrc\n",
1708 rcVBox));
1709 rc = -RTErrConvertToErrno(rcVBox);
1710 }
1711 }
1712
1713 /* get ISR */
1714 if (!rc)
1715 {
1716 rc = request_irq(pcidev->irq, vboxadd_irq_handler,
1717#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 20)
1718 IRQF_SHARED,
1719#else
1720 SA_SHIRQ,
1721#endif
1722 VBOXADD_NAME, vboxDev);
1723 if (rc)
1724 LogRel(("vboxadd: could not request IRQ %d, err: %d\n", pcidev->irq, rc));
1725 else
1726 vboxDev->irq = pcidev->irq;
1727 }
1728
1729 if (!rc)
1730 {
1731 VBoxGuestFilterMaskInfo info;
1732
1733 init_waitqueue_head (&vboxDev->eventq);
1734 /* Register for notification when the host absolute pointer position
1735 * changes. */
1736 info.u32OrMask = VMMDEV_EVENT_MOUSE_POSITION_CHANGED;
1737 info.u32NotMask = 0;
1738 rcVBox = vboxadd_control_filter_mask(&info);
1739 if (RT_FAILURE(rcVBox))
1740 {
1741 LogRel(("vboxadd: failed to register for VMMDEV_EVENT_MOUSE_POSITION_CHANGED events: %Rrc\n",
1742 rcVBox));
1743 rc = -RTErrConvertToErrno(rcVBox);
1744 }
1745 }
1746
1747 if (!rc)
1748 {
1749 /* some useful information for the user but don't show this on the console */
1750 LogRel(("VirtualBox device settings: major %d, IRQ %d, "
1751 "I/O port 0x%x, MMIO at 0x%x (size 0x%x), "
1752 "hypervisor window at 0x%p (size 0x%x)\n",
1753 vbox_major, vboxDev->irq, vboxDev->io_port,
1754 vboxDev->vmmdevmem, vboxDev->vmmdevmem_size,
1755 vboxDev->hypervisorStart, vboxDev->hypervisorSize));
1756 printk(KERN_DEBUG "vboxadd: Successfully loaded version "
1757 VBOX_VERSION_STRING " (interface " xstr(VMMDEV_VERSION) ")\n");
1758 }
1759 else /* Clean up on failure */
1760 {
1761 if (fHaveGuestLib)
1762 VbglTerminate();
1763 if (vboxDev)
1764 free_resources();
1765 if (fHaveVBoxUser)
1766 misc_deregister(&gMiscVBoxUser);
1767 if (fHaveVBoxAdd && vbox_major > 0)
1768 unregister_chrdev(vbox_major, VBOXADD_NAME);
1769 else if (fHaveVBoxAdd)
1770 misc_deregister(&gMiscVBoxAdd);
1771 }
1772
1773 /* We always release this. Presumably because we no longer need to do
1774 * anything with the device structure. */
1775 if (pcidev)
1776 PCI_DEV_PUT(pcidev);
1777
1778 return rc;
1779}
1780
1781/**
1782 * Module termination
1783 *
1784 */
1785static __exit void fini(void)
1786{
1787 misc_deregister(&gMiscVBoxUser);
1788 if (vbox_major > 0)
1789 unregister_chrdev(vbox_major, VBOXADD_NAME);
1790 else
1791 misc_deregister(&gMiscVBoxAdd);
1792 free_resources();
1793 vboxadd_cmc_fini ();
1794}
1795
1796module_init(init);
1797module_exit(fini);
1798
1799/* PCI hotplug structure */
1800static const struct pci_device_id __devinitdata vmmdev_pci_id[] =
1801{
1802 {
1803 .vendor = VMMDEV_VENDORID,
1804 .device = VMMDEV_DEVICEID
1805 },
1806 {
1807 /* empty entry */
1808 }
1809};
1810MODULE_DEVICE_TABLE(pci, vmmdev_pci_id);
1811
1812
1813
1814/*
1815 * Local Variables:
1816 * c-mode: bsd
1817 * indent-tabs-mode: nil
1818 * c-plusplus: evil
1819 * End:
1820 */
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