VirtualBox

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

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

vboxmod.c: Fixed AssertReturn, must recheck cb.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 57.9 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#ifdef DEBUG /* Only allow random user applications to spam the log in
965 * debug additions builds */
966 if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_LOG(0))
967 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
968 {
969 char *pszMessage;
970
971 IOCTL_LOG_ENTRY(arg);
972 pszMessage = kmalloc(_IOC_SIZE(cmd), GFP_KERNEL);
973 if (NULL == pszMessage)
974 {
975 LogRelFunc(("VBOXGUEST_IOCTL_LOG: cannot allocate %d bytes of memory!\n",
976 _IOC_SIZE(cmd)));
977 rc = -ENOMEM;
978 }
979 if ( (0 == rc)
980 && copy_from_user(pszMessage, (void*)arg, _IOC_SIZE(cmd)))
981 {
982 LogRelFunc(("VBOXGUEST_IOCTL_LOG: copy_from_user failed!\n"));
983 rc = -EFAULT;
984 }
985 if (0 == rc)
986 {
987 Log(("%.*s", _IOC_SIZE(cmd), pszMessage));
988 }
989 if (NULL != pszMessage)
990 {
991 kfree(pszMessage);
992 }
993 IOCTL_LOG_EXIT(arg);
994 }
995 else
996#endif
997 if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_VMMREQUEST(0))
998 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
999 {
1000 VMMDevRequestHeader reqHeader;
1001 VMMDevRequestHeader *reqFull = NULL;
1002 size_t cbRequestSize;
1003 size_t cbVanillaRequestSize;
1004
1005 IOCTL_VMM_ENTRY(arg);
1006 if (copy_from_user(&reqHeader, (void*)arg, sizeof(reqHeader)))
1007 {
1008 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: copy_from_user failed for vmm request!\n"));
1009 rc = -EFAULT;
1010 }
1011 if (0 == rc)
1012 {
1013 /* get the request size */
1014 cbVanillaRequestSize = vmmdevGetRequestSize(reqHeader.requestType);
1015 if (!cbVanillaRequestSize)
1016 {
1017 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: invalid request type: %d\n",
1018 reqHeader.requestType));
1019 rc = -EINVAL;
1020 }
1021 }
1022 if (0 == rc)
1023 {
1024 cbRequestSize = reqHeader.size;
1025 if (cbRequestSize < cbVanillaRequestSize)
1026 {
1027 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: invalid request size: %d min: %d type: %d\n",
1028 cbRequestSize,
1029 cbVanillaRequestSize,
1030 reqHeader.requestType));
1031 rc = -EINVAL;
1032 }
1033 }
1034 if (0 == rc)
1035 {
1036 /* request storage for the full request */
1037 rc = VbglGRAlloc(&reqFull, cbRequestSize, reqHeader.requestType);
1038 if (RT_FAILURE(rc))
1039 {
1040 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: could not allocate request structure! rc = %d\n", rc));
1041 rc = -EFAULT;
1042 }
1043 }
1044 if (0 == rc)
1045 {
1046 /* now get the full request */
1047 if (copy_from_user(reqFull, (void*)arg, cbRequestSize))
1048 {
1049 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: failed to fetch full request from user space!\n"));
1050 rc = -EFAULT;
1051 }
1052 }
1053
1054 /* now issue the request */
1055 if (0 == rc)
1056 {
1057 int rrc = VbglGRPerform(reqFull);
1058
1059 /* asynchronous processing? */
1060 if (rrc == VINF_HGCM_ASYNC_EXECUTE)
1061 {
1062 VMMDevHGCMRequestHeader *reqHGCM = (VMMDevHGCMRequestHeader*)reqFull;
1063 wait_event_interruptible (vboxDev->eventq, reqHGCM->fu32Flags & VBOX_HGCM_REQ_DONE);
1064 rrc = reqFull->rc;
1065 }
1066
1067 /* failed? */
1068 if (RT_FAILURE(rrc) || RT_FAILURE(reqFull->rc))
1069 {
1070 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: request execution failed!\n"));
1071 rc = RT_FAILURE(rrc) ? -RTErrConvertToErrno(rrc)
1072 : -RTErrConvertToErrno(reqFull->rc);
1073 }
1074 else
1075 {
1076 /* success, copy the result data to user space */
1077 if (copy_to_user((void*)arg, (void*)reqFull, cbRequestSize))
1078 {
1079 LogRelFunc(("VBOXGUEST_IOCTL_VMMREQUEST: error copying request result to user space!\n"));
1080 rc = -EFAULT;
1081 }
1082 }
1083 }
1084 if (NULL != reqFull)
1085 VbglGRFree(reqFull);
1086 IOCTL_VMM_EXIT(arg);
1087 }
1088 else if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_HGCM_CALL(0))
1089 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
1090 {
1091 /* Do the HGCM call using the Vbgl bits */
1092 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CALL", arg);
1093 rc = vboxadd_hgcm_call(arg, _IOC_SIZE(cmd));
1094 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CALL", arg);
1095 }
1096 else if ( VBOXGUEST_IOCTL_STRIP_SIZE(VBOXGUEST_IOCTL_HGCM_CALL_TIMED(0))
1097 == VBOXGUEST_IOCTL_STRIP_SIZE(cmd))
1098 {
1099 /* Do the HGCM call using the Vbgl bits */
1100 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CALL_TIMED", arg);
1101 rc = vboxadd_hgcm_call_timed(arg, _IOC_SIZE(cmd));
1102 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CALL_TIMED", arg);
1103 }
1104 else
1105 {
1106 switch (cmd)
1107 {
1108 case VBOXGUEST_IOCTL_WAITEVENT:
1109 IOCTL_ENTRY("VBOXGUEST_IOCTL_WAITEVENT", arg);
1110 rc = vboxadd_wait_event((void *) arg);
1111 IOCTL_EXIT("VBOXGUEST_IOCTL_WAITEVENT", arg);
1112 break;
1113 case VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS:
1114 IOCTL_ENTRY("VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS", arg);
1115 ++vboxDev->u32GuestInterruptions;
1116 IOCTL_EXIT("VBOXGUEST_IOCTL_CANCEL_ALL_WAITEVENTS", arg);
1117 break;
1118 case VBOXGUEST_IOCTL_HGCM_CONNECT:
1119 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_CONNECT", arg);
1120 rc = vboxadd_hgcm_connect(filp, arg);
1121 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_CONNECT", arg);
1122 break;
1123 case VBOXGUEST_IOCTL_HGCM_DISCONNECT:
1124 IOCTL_ENTRY("VBOXGUEST_IOCTL_HGCM_DISCONNECT", arg);
1125 vboxadd_hgcm_disconnect(filp, arg);
1126 IOCTL_EXIT("VBOXGUEST_IOCTL_HGCM_DISCONNECT", arg);
1127 break;
1128 case VBOXGUEST_IOCTL_CTL_FILTER_MASK:
1129 {
1130 VBoxGuestFilterMaskInfo info;
1131 IOCTL_ENTRY("VBOXGUEST_IOCTL_CTL_FILTER_MASK", arg);
1132 if (copy_from_user((void*)&info, (void*)arg, sizeof(info)))
1133 {
1134 LogRelFunc(("VBOXGUEST_IOCTL_CTL_FILTER_MASK: error getting parameters from user space!\n"));
1135 rc = -EFAULT;
1136 break;
1137 }
1138 rc = -RTErrConvertToErrno(vboxadd_control_filter_mask(&info));
1139 IOCTL_EXIT("VBOXGUEST_IOCTL_CTL_FILTER_MASK", arg);
1140 break;
1141 }
1142 default:
1143 LogRelFunc(("unknown command: %x\n", cmd));
1144 rc = -EINVAL;
1145 break;
1146 }
1147 }
1148 return rc;
1149}
1150
1151/**
1152 * Poll function. This returns "ready to read" if the guest is in absolute
1153 * mouse pointer mode and the pointer position has changed since the last
1154 * poll.
1155 */
1156unsigned int
1157vboxadd_poll (struct file *file, poll_table *wait)
1158{
1159 int result = 0;
1160 poll_wait(file, &vboxDev->eventq, wait);
1161 if (vboxDev->u32Events & VMMDEV_EVENT_MOUSE_POSITION_CHANGED)
1162 result = (POLLIN | POLLRDNORM);
1163 vboxDev->u32Events &= ~VMMDEV_EVENT_MOUSE_POSITION_CHANGED;
1164 return result;
1165}
1166
1167/** Asynchronous notification activation method. */
1168static int
1169vboxadd_fasync(int fd, struct file *file, int mode)
1170{
1171 return fasync_helper(fd, file, mode, &vboxDev->async_queue);
1172}
1173
1174/**
1175 * Dummy read function - we only supply this because we implement poll and
1176 * fasync.
1177 */
1178static ssize_t
1179vboxadd_read (struct file *file, char *buf, size_t count, loff_t *loff)
1180{
1181 if (0 == count || *loff != 0)
1182 {
1183 return -EINVAL;
1184 }
1185 buf[0] = 0;
1186 return 1;
1187}
1188
1189/**
1190 * File close handler for vboxadd. Clean up any HGCM connections associated
1191 * with the open file which might still be open.
1192 */
1193static int vboxadd_release(struct inode *inode, struct file * filp)
1194{
1195 vboxadd_unregister_all_hgcm_connections(filp);
1196 /* Deactivate our asynchronous queue. */
1197 vboxadd_fasync(-1, filp, 0);
1198 return 0;
1199}
1200
1201/**
1202 * File close handler for vboxuser. Clean up any HGCM connections associated
1203 * with the open file which might still be open.
1204 */
1205static int vboxuser_release(struct inode *inode, struct file * filp)
1206{
1207 vboxadd_unregister_all_hgcm_connections(filp);
1208 return 0;
1209}
1210
1211/** file operations for the vboxadd device */
1212static struct file_operations vboxadd_fops =
1213{
1214 .owner = THIS_MODULE,
1215 .open = vboxadd_open,
1216 .ioctl = vboxadd_ioctl,
1217 .poll = vboxadd_poll,
1218 .fasync = vboxadd_fasync,
1219 .read = vboxadd_read,
1220 .release = vboxadd_release,
1221 .llseek = no_llseek
1222};
1223
1224/** Miscellaneous device allocation for vboxadd */
1225static struct miscdevice gMiscVBoxAdd =
1226{
1227 minor: MISC_DYNAMIC_MINOR,
1228 name: VBOXADD_NAME,
1229 fops: &vboxadd_fops
1230};
1231
1232/** file operations for the vboxuser device */
1233static struct file_operations vboxuser_fops =
1234{
1235 .owner = THIS_MODULE,
1236 .open = vboxadd_open,
1237 .ioctl = vboxuser_ioctl,
1238 .release = vboxuser_release,
1239 .llseek = no_llseek
1240};
1241
1242/** Miscellaneous device allocation for vboxuser */
1243static struct miscdevice gMiscVBoxUser =
1244{
1245 minor: MISC_DYNAMIC_MINOR,
1246 name: VBOXUSER_NAME,
1247 fops: &vboxuser_fops
1248};
1249
1250#ifndef IRQ_RETVAL
1251/* interrupt handlers in 2.4 kernels don't return anything */
1252# define irqreturn_t void
1253# define IRQ_RETVAL(n)
1254#endif
1255
1256/**
1257 * vboxadd_irq_handler
1258 *
1259 * Interrupt handler
1260 *
1261 * @returns scsi error code
1262 * @param irq Irq number
1263 * @param dev_id Irq handler parameter
1264 * @param regs Regs
1265 *
1266 */
1267#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 19)
1268static irqreturn_t vboxadd_irq_handler(int irq, void *dev_id)
1269#else
1270static irqreturn_t vboxadd_irq_handler(int irq, void *dev_id, struct pt_regs *regs)
1271#endif
1272{
1273 int fIRQTaken = 0;
1274 int rcVBox;
1275
1276#ifdef IRQ_DEBUG
1277 Log(("vboxadd IRQ_DEBUG: vboxDev->pVMMDevMemory=%p vboxDev->pVMMDevMemory->fHaveEvents=%d\n",
1278 vboxDev->pVMMDevMemory, vboxDev->pVMMDevMemory->V.V1_04.fHaveEvents));
1279#endif
1280
1281 /* check if IRQ was asserted by VBox */
1282 if (vboxDev->pVMMDevMemory->V.V1_04.fHaveEvents != 0)
1283 {
1284#ifdef IRQ_DEBUG
1285 Log(("vboxadd IRQ_DEBUG: got IRQ with event mask 0x%x\n",
1286 vboxDev->irqAckRequest->events));
1287#endif
1288
1289 /* make a copy of the event mask */
1290 rcVBox = VbglGRPerform (&vboxDev->irqAckRequest->header);
1291 if (RT_SUCCESS(rcVBox) && RT_SUCCESS(vboxDev->irqAckRequest->header.rc))
1292 {
1293 if (RT_LIKELY (vboxDev->irqAckRequest->events))
1294 {
1295 vboxDev->u32Events |= vboxDev->irqAckRequest->events;
1296 if ( vboxDev->irqAckRequest->events
1297 & VMMDEV_EVENT_MOUSE_POSITION_CHANGED)
1298 kill_fasync(&vboxDev->async_queue, SIGIO, POLL_IN);
1299 wake_up (&vboxDev->eventq);
1300 }
1301 }
1302 else
1303 {
1304 /* impossible... */
1305 LogRelFunc(("IRQ was not acknowledged! rc = %Rrc, header.rc = %Rrc\n",
1306 rcVBox, vboxDev->irqAckRequest->header.rc));
1307 BUG ();
1308 }
1309
1310 /* it was ours! */
1311 fIRQTaken = 1;
1312 }
1313#ifdef IRQ_DEBUG
1314 else
1315 {
1316 /* we might be attached to a shared interrupt together with another device. */
1317 Log(("vboxadd IRQ_DEBUG: stale IRQ mem=%p events=%d devevents=%#x\n",
1318 vboxDev->pVMMDevMemory,
1319 vboxDev->pVMMDevMemory->V.V1_04.fHaveEvents,
1320 vboxDev->u32Events));
1321 }
1322#endif
1323 /* it was ours */
1324 return IRQ_RETVAL(fIRQTaken);
1325}
1326
1327/**
1328 * Helper function to reserve a fixed kernel address space window
1329 * and tell the VMM that it can safely put its hypervisor there.
1330 * This function might fail which is not a critical error.
1331 */
1332static int vboxadd_reserve_hypervisor(void)
1333{
1334 VMMDevReqHypervisorInfo *req = NULL;
1335 int rcVBox;
1336
1337 /* allocate request structure */
1338 rcVBox = VbglGRAlloc(
1339 (VMMDevRequestHeader**)&req,
1340 sizeof(VMMDevReqHypervisorInfo),
1341 VMMDevReq_GetHypervisorInfo
1342 );
1343 if (RT_FAILURE(rcVBox))
1344 {
1345 LogRelFunc(("failed to allocate hypervisor info structure! rc = %Rrc\n", rcVBox));
1346 goto bail_out;
1347 }
1348 /* query the hypervisor information */
1349 rcVBox = VbglGRPerform(&req->header);
1350 if (RT_SUCCESS(rcVBox) && RT_SUCCESS(req->header.rc))
1351 {
1352 /* are we supposed to make a reservation? */
1353 if (req->hypervisorSize)
1354 {
1355 /** @todo repeat this several times until we get an address the host likes */
1356
1357 void *hypervisorArea;
1358 /* reserve another 4MB because the start needs to be 4MB aligned */
1359 uint32_t hypervisorSize = req->hypervisorSize + 0x400000;
1360 /* perform a fictive IO space mapping */
1361 hypervisorArea = ioremap(HYPERVISOR_PHYSICAL_START, hypervisorSize);
1362 if (hypervisorArea)
1363 {
1364 /* communicate result to VMM, align at 4MB */
1365 req->hypervisorStart = (VMMDEVHYPPTR32)(uintptr_t)RT_ALIGN_P(hypervisorArea, 0x400000);
1366 req->header.requestType = VMMDevReq_SetHypervisorInfo;
1367 req->header.rc = VERR_GENERAL_FAILURE;
1368 rcVBox = VbglGRPerform(&req->header);
1369 if (RT_SUCCESS(rcVBox) && RT_SUCCESS(req->header.rc))
1370 {
1371 /* store mapping for future unmapping */
1372 vboxDev->hypervisorStart = hypervisorArea;
1373 vboxDev->hypervisorSize = hypervisorSize;
1374 }
1375 else
1376 {
1377 LogRelFunc(("failed to set hypervisor region! rc = %Rrc, header.rc = %Rrc\n",
1378 rcVBox, req->header.rc));
1379 goto bail_out;
1380 }
1381 }
1382 else
1383 {
1384 LogRelFunc(("failed to allocate 0x%x bytes of IO space\n", hypervisorSize));
1385 goto bail_out;
1386 }
1387 }
1388 }
1389 else
1390 {
1391 LogRelFunc(("failed to query hypervisor info! rc = %Rrc, header.rc = %Rrc\n",
1392 rcVBox, req->header.rc));
1393 goto bail_out;
1394 }
1395 /* successful return */
1396 VbglGRFree(&req->header);
1397 return 0;
1398
1399bail_out:
1400 /* error return */
1401 if (req)
1402 VbglGRFree(&req->header);
1403 return 1;
1404}
1405
1406/**
1407 * Helper function to free the hypervisor address window
1408 *
1409 */
1410static int vboxadd_free_hypervisor(void)
1411{
1412 VMMDevReqHypervisorInfo *req = NULL;
1413 int rcVBox;
1414
1415 /* allocate request structure */
1416 rcVBox = VbglGRAlloc(
1417 (VMMDevRequestHeader**)&req,
1418 sizeof(VMMDevReqHypervisorInfo),
1419 VMMDevReq_SetHypervisorInfo
1420 );
1421 if (RT_FAILURE(rcVBox))
1422 {
1423 LogRelFunc(("failed to allocate hypervisor info structure! rc = %Rrc\n", rcVBox));
1424 goto bail_out;
1425 }
1426 /* reset the hypervisor information */
1427 req->hypervisorStart = 0;
1428 req->hypervisorSize = 0;
1429 rcVBox = VbglGRPerform(&req->header);
1430 if (RT_SUCCESS(rcVBox) && RT_SUCCESS(req->header.rc))
1431 {
1432 /* now we can free the associated IO space mapping */
1433 iounmap(vboxDev->hypervisorStart);
1434 vboxDev->hypervisorStart = 0;
1435 }
1436 else
1437 {
1438 LogRelFunc(("failed to reset hypervisor info! rc = %Rrc, header.rc = %Rrc\n",
1439 rcVBox, req->header.rc));
1440 goto bail_out;
1441 }
1442 return 0;
1443
1444 bail_out:
1445 if (req)
1446 VbglGRFree(&req->header);
1447 return 1;
1448}
1449
1450/**
1451 * Helper to free resources
1452 *
1453 */
1454static void free_resources(void)
1455{
1456 if (vboxDev)
1457 {
1458 {
1459 /* Unregister notifications when the host absolute pointer
1460 * position changes. */
1461 VBoxGuestFilterMaskInfo info;
1462 info.u32OrMask = 0;
1463 info.u32NotMask = VMMDEV_EVENT_MOUSE_POSITION_CHANGED;
1464 vboxadd_control_filter_mask(&info);
1465 }
1466 /* Detach from IRQ before cleaning up! */
1467 if (vboxDev->irq)
1468 free_irq(vboxDev->irq, vboxDev);
1469 if (vboxDev->hypervisorStart)
1470 vboxadd_free_hypervisor();
1471 if (vboxDev->irqAckRequest)
1472 {
1473 VbglGRFree(&vboxDev->irqAckRequest->header);
1474 VbglTerminate();
1475 }
1476 if (vboxDev->pVMMDevMemory)
1477 iounmap(vboxDev->pVMMDevMemory);
1478 if (vboxDev->vmmdevmem)
1479 release_mem_region(vboxDev->vmmdevmem, vboxDev->vmmdevmem_size);
1480 kfree(vboxDev);
1481 vboxDev = NULL;
1482 }
1483}
1484
1485#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 20)
1486#define PCI_DEV_GET(v,d,p) pci_get_device(v,d,p)
1487#define PCI_DEV_PUT(x) pci_dev_put(x)
1488#else
1489#define PCI_DEV_GET(v,d,p) pci_find_device(v,d,p)
1490#define PCI_DEV_PUT(x) do {} while(0)
1491#endif
1492
1493/**
1494 * Module initialization
1495 *
1496 */
1497static __init int init(void)
1498{
1499 int rc = 0, rcVBox = VINF_SUCCESS;
1500 bool fHaveVBoxAdd = false, fHaveVBoxUser = false, fHaveGuestLib = false;
1501 struct pci_dev *pcidev = NULL;
1502
1503 rcVBox = vboxadd_cmc_init();
1504 if (RT_FAILURE(rcVBox))
1505 {
1506 printk (KERN_ERR "vboxadd: could not init cmc, VBox error code %d.\n", rcVBox);
1507 rc = -RTErrConvertToErrno(rcVBox);
1508 }
1509
1510 /* Detect PCI device */
1511 if (!rc)
1512 {
1513 pcidev = PCI_DEV_GET(VMMDEV_VENDORID, VMMDEV_DEVICEID, pcidev);
1514 if (!pcidev)
1515 {
1516 printk(KERN_ERR "vboxadd: VirtualBox Guest PCI device not found.\n");
1517 rc = -ENODEV;
1518 }
1519 }
1520
1521 if (!rc)
1522 {
1523 rc = pci_enable_device (pcidev);
1524 if (rc)
1525 LogRel(("vboxadd: could not enable device: %d\n", rc));
1526 }
1527 if (!rc)
1528 LogRel(("Starting VirtualBox version %s Guest Additions\n",
1529 VBOX_VERSION_STRING));
1530
1531 /* Register vboxadd */
1532 if (!rc && vbox_major > 0) /* Register as a character device in this case */
1533 {
1534 rc = register_chrdev(vbox_major, VBOXADD_NAME, &vboxadd_fops);
1535 if (rc) /* As we pass a non-zero major, rc should be zero on success. */
1536 LogRel(("vboxadd: register_chrdev failed: vbox_major: %d, err = %d\n",
1537 vbox_major, rc));
1538 }
1539 else if (!rc) /* Register as a miscellaneous device otherwise */
1540 {
1541 rc = misc_register(&gMiscVBoxAdd);
1542 if (rc)
1543 LogRel(("vboxadd: misc_register failed for %s (rc=%d)\n",
1544 VBOXADD_NAME, rc));
1545 }
1546 if (!rc)
1547 fHaveVBoxAdd = true;
1548
1549 /* Register our user session device */
1550 if (!rc)
1551 {
1552 rc = misc_register(&gMiscVBoxUser);
1553 if (rc)
1554 LogRel(("vboxadd: misc_register failed for %s (rc=%d)\n",
1555 VBOXUSER_NAME, rc));
1556 }
1557 if (!rc)
1558 fHaveVBoxUser = true;
1559
1560 /* allocate and initialize device extension */
1561 if (!rc)
1562 {
1563 vboxDev = kmalloc(sizeof(*vboxDev), GFP_KERNEL);
1564 if (vboxDev)
1565 memset(vboxDev, 0, sizeof(*vboxDev));
1566 else
1567 {
1568 LogRel(("vboxadd: could not allocate private device structure\n"));
1569 rc = -ENOMEM;
1570 }
1571 }
1572
1573 if (!rc)
1574 {
1575 /* get the IO port region */
1576 vboxDev->io_port = pci_resource_start(pcidev, 0);
1577
1578 /* get the memory region */
1579 vboxDev->vmmdevmem = pci_resource_start(pcidev, 1);
1580 vboxDev->vmmdevmem_size = pci_resource_len(pcidev, 1);
1581
1582 /* all resources found? */
1583 if (!vboxDev->io_port || !vboxDev->vmmdevmem || !vboxDev->vmmdevmem_size)
1584 {
1585 LogRel(("vboxadd: did not find expected hardware resources\n"));
1586 rc = -ENXIO;
1587 }
1588 }
1589
1590 /* request ownership of adapter memory */
1591 if (!rc && !request_mem_region(vboxDev->vmmdevmem, vboxDev->vmmdevmem_size,
1592 VBOXADD_NAME))
1593 {
1594 LogRel(("vboxadd: failed to obtain adapter memory\n"));
1595 rc = -EBUSY;
1596 }
1597
1598 /* map adapter memory into kernel address space and check version */
1599 if (!rc)
1600 {
1601 vboxDev->pVMMDevMemory = (VMMDevMemory *) ioremap(vboxDev->vmmdevmem,
1602 vboxDev->vmmdevmem_size);
1603 if (!vboxDev->pVMMDevMemory)
1604 {
1605 LogRel(("vboxadd: ioremap failed\n"));
1606 rc = -ENOMEM;
1607 }
1608 }
1609
1610 if (!rc && (vboxDev->pVMMDevMemory->u32Version != VMMDEV_MEMORY_VERSION))
1611 {
1612 LogRel(("vboxadd: invalid VMM device memory version! (got 0x%x, expected 0x%x)\n",
1613 vboxDev->pVMMDevMemory->u32Version, VMMDEV_MEMORY_VERSION));
1614 rc = -ENXIO;
1615 }
1616
1617 /* initialize ring 0 guest library */
1618 if (!rc)
1619 {
1620 rcVBox = VbglInit(vboxDev->io_port, vboxDev->pVMMDevMemory);
1621 if (RT_FAILURE(rcVBox))
1622 {
1623 LogRel(("vboxadd: could not initialize VBGL subsystem: %Rrc\n",
1624 rcVBox));
1625 rc = -RTErrConvertToErrno(rcVBox);
1626 }
1627 }
1628 if (!rc)
1629 fHaveGuestLib = true;
1630
1631 /* report guest information to host, this must be done as the very first request */
1632 if (!rc)
1633 {
1634 VMMDevReportGuestInfo *infoReq = NULL;
1635
1636 rcVBox = VbglGRAlloc((VMMDevRequestHeader**)&infoReq,
1637 sizeof(VMMDevReportGuestInfo), VMMDevReq_ReportGuestInfo);
1638 if (RT_FAILURE(rcVBox))
1639 {
1640 LogRel(("vboxadd: could not allocate request structure: %Rrc\n", rcVBox));
1641 rc = -RTErrConvertToErrno(rcVBox);
1642 }
1643 /* report guest version to host, the VMMDev requires that to be done
1644 * before any other VMMDev operations. */
1645 if (infoReq)
1646 {
1647 infoReq->guestInfo.additionsVersion = VMMDEV_VERSION;
1648#if LINUX_VERSION_CODE > KERNEL_VERSION(2, 6, 0)
1649 infoReq->guestInfo.osType = VBOXOSTYPE_Linux26;
1650#else
1651 infoReq->guestInfo.osType = VBOXOSTYPE_Linux24;
1652#endif
1653 rcVBox = VbglGRPerform(&infoReq->header);
1654 }
1655 if ( infoReq
1656 && ( RT_FAILURE(rcVBox)
1657 || RT_FAILURE(infoReq->header.rc)))
1658 {
1659 LogRel(("vboxadd: error reporting guest information to host: %Rrc, header: %Rrc\n",
1660 rcVBox, infoReq->header.rc));
1661 rc = RT_FAILURE(rcVBox) ? -RTErrConvertToErrno(rcVBox)
1662 : -RTErrConvertToErrno(infoReq->header.rc);
1663 }
1664 if (infoReq)
1665 VbglGRFree(&infoReq->header);
1666 }
1667
1668 /* Unset the graphics capability until/unless X is loaded. */
1669 /** @todo check the error code once we bump the additions version.
1670 For now we ignore it for compatibility with older hosts. */
1671 if (!rc)
1672 {
1673 VMMDevReqGuestCapabilities2 *vmmreqGuestCaps;
1674
1675 rcVBox = VbglGRAlloc((VMMDevRequestHeader**)&vmmreqGuestCaps,
1676 sizeof(VMMDevReqGuestCapabilities2),
1677 VMMDevReq_SetGuestCapabilities);
1678 if (RT_FAILURE(rcVBox))
1679 {
1680 LogRel(("vboxadd: could not allocate request structure: %Rrc\n",
1681 rcVBox));
1682 rc = -RTErrConvertToErrno(rcVBox);
1683 }
1684 else
1685 {
1686 vmmreqGuestCaps->u32OrMask = 0;
1687 vmmreqGuestCaps->u32NotMask = VMMDEV_GUEST_SUPPORTS_GRAPHICS;
1688 rcVBox = VbglGRPerform(&vmmreqGuestCaps->header);
1689 if (RT_FAILURE(rcVBox))
1690 {
1691 LogRel(("vboxadd: could not allocate request structure: %Rrc\n",
1692 rcVBox));
1693 rc = -RTErrConvertToErrno(rcVBox);
1694 }
1695 VbglGRFree(&vmmreqGuestCaps->header);
1696 }
1697 }
1698
1699 /* perform hypervisor address space reservation */
1700 if (!rc && vboxadd_reserve_hypervisor())
1701 {
1702 /* we just ignore the error, no address window reservation, non fatal */
1703 }
1704
1705 /* allocate a VMM request structure for use in the ISR */
1706 if (!rc)
1707 {
1708 rcVBox = VbglGRAlloc((VMMDevRequestHeader**)&vboxDev->irqAckRequest,
1709 sizeof(VMMDevEvents), VMMDevReq_AcknowledgeEvents);
1710 if (RT_FAILURE(rcVBox))
1711 {
1712 LogRel(("vboxadd: could not allocate request structure: %Rrc\n",
1713 rcVBox));
1714 rc = -RTErrConvertToErrno(rcVBox);
1715 }
1716 }
1717
1718 /* get ISR */
1719 if (!rc)
1720 {
1721 rc = request_irq(pcidev->irq, vboxadd_irq_handler,
1722#if LINUX_VERSION_CODE >= KERNEL_VERSION(2, 6, 20)
1723 IRQF_SHARED,
1724#else
1725 SA_SHIRQ,
1726#endif
1727 VBOXADD_NAME, vboxDev);
1728 if (rc)
1729 LogRel(("vboxadd: could not request IRQ %d, err: %d\n", pcidev->irq, rc));
1730 else
1731 vboxDev->irq = pcidev->irq;
1732 }
1733
1734 if (!rc)
1735 {
1736 VBoxGuestFilterMaskInfo info;
1737
1738 init_waitqueue_head (&vboxDev->eventq);
1739 /* Register for notification when the host absolute pointer position
1740 * changes. */
1741 info.u32OrMask = VMMDEV_EVENT_MOUSE_POSITION_CHANGED;
1742 info.u32NotMask = 0;
1743 rcVBox = vboxadd_control_filter_mask(&info);
1744 if (RT_FAILURE(rcVBox))
1745 {
1746 LogRel(("vboxadd: failed to register for VMMDEV_EVENT_MOUSE_POSITION_CHANGED events: %Rrc\n",
1747 rcVBox));
1748 rc = -RTErrConvertToErrno(rcVBox);
1749 }
1750 }
1751
1752 if (!rc)
1753 {
1754 /* some useful information for the user but don't show this on the console */
1755 LogRel(("VirtualBox device settings: major %d, IRQ %d, "
1756 "I/O port 0x%x, MMIO at 0x%x (size 0x%x), "
1757 "hypervisor window at 0x%p (size 0x%x)\n",
1758 vbox_major, vboxDev->irq, vboxDev->io_port,
1759 vboxDev->vmmdevmem, vboxDev->vmmdevmem_size,
1760 vboxDev->hypervisorStart, vboxDev->hypervisorSize));
1761 printk(KERN_DEBUG "vboxadd: Successfully loaded version "
1762 VBOX_VERSION_STRING " (interface " xstr(VMMDEV_VERSION) ")\n");
1763 }
1764 else /* Clean up on failure */
1765 {
1766 if (fHaveGuestLib)
1767 VbglTerminate();
1768 if (vboxDev)
1769 free_resources();
1770 if (fHaveVBoxUser)
1771 misc_deregister(&gMiscVBoxUser);
1772 if (fHaveVBoxAdd && vbox_major > 0)
1773 unregister_chrdev(vbox_major, VBOXADD_NAME);
1774 else if (fHaveVBoxAdd)
1775 misc_deregister(&gMiscVBoxAdd);
1776 }
1777
1778 /* We always release this. Presumably because we no longer need to do
1779 * anything with the device structure. */
1780 if (pcidev)
1781 PCI_DEV_PUT(pcidev);
1782
1783 return rc;
1784}
1785
1786/**
1787 * Module termination
1788 *
1789 */
1790static __exit void fini(void)
1791{
1792 misc_deregister(&gMiscVBoxUser);
1793 if (vbox_major > 0)
1794 unregister_chrdev(vbox_major, VBOXADD_NAME);
1795 else
1796 misc_deregister(&gMiscVBoxAdd);
1797 free_resources();
1798 vboxadd_cmc_fini ();
1799}
1800
1801module_init(init);
1802module_exit(fini);
1803
1804/* PCI hotplug structure */
1805static const struct pci_device_id __devinitdata vmmdev_pci_id[] =
1806{
1807 {
1808 .vendor = VMMDEV_VENDORID,
1809 .device = VMMDEV_DEVICEID
1810 },
1811 {
1812 /* empty entry */
1813 }
1814};
1815MODULE_DEVICE_TABLE(pci, vmmdev_pci_id);
1816
1817
1818
1819/*
1820 * Local Variables:
1821 * c-mode: bsd
1822 * indent-tabs-mode: nil
1823 * c-plusplus: evil
1824 * End:
1825 */
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