VirtualBox

source: vbox/trunk/src/VBox/Additions/common/VBoxGuest/VBoxGuest-haiku.c@ 46223

Last change on this file since 46223 was 43411, checked in by vboxsync, 12 years ago

Additions/haiku: cleanup.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.2 KB
Line 
1/* $Id: VBoxGuest-haiku.c 43411 2012-09-22 18:42:16Z vboxsync $ */
2/** @file
3 * VBoxGuest kernel module, Haiku Guest Additions, implementation.
4 */
5
6/*
7 * Copyright (C) 2012 Oracle Corporation
8 *
9 * This file is part of VirtualBox Open Source Edition (OSE), as
10 * available from http://www.virtualbox.org. This file is free software;
11 * you can redistribute it and/or modify it under the terms of the GNU
12 * General Public License (GPL) as published by the Free Software
13 * Foundation, in version 2 as it comes in the "COPYING" file of the
14 * VirtualBox OSE distribution. VirtualBox OSE is distributed in the
15 * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
16 */
17
18/*
19 * This code is based on:
20 *
21 * VirtualBox Guest Additions for Haiku.
22 * Copyright (c) 2011 Mike Smith <[email protected]>
23 * François Revol <[email protected]>
24 *
25 * Permission is hereby granted, free of charge, to any person
26 * obtaining a copy of this software and associated documentation
27 * files (the "Software"), to deal in the Software without
28 * restriction, including without limitation the rights to use,
29 * copy, modify, merge, publish, distribute, sublicense, and/or sell
30 * copies of the Software, and to permit persons to whom the
31 * Software is furnished to do so, subject to the following
32 * conditions:
33 *
34 * The above copyright notice and this permission notice shall be
35 * included in all copies or substantial portions of the Software.
36 *
37 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
38 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
39 * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
40 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
41 * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
42 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
43 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
44 * OTHER DEALINGS IN THE SOFTWARE.
45 */
46
47/*******************************************************************************
48* Header Files *
49*******************************************************************************/
50#define IN_VBOXGUEST
51#include <sys/param.h>
52#include <sys/types.h>
53#include <sys/uio.h>
54#include <OS.h>
55#include <Drivers.h>
56#include <KernelExport.h>
57#include <PCI.h>
58
59#include "VBoxGuest-haiku.h"
60#include "VBoxGuestInternal.h"
61#include <VBox/log.h>
62#include <iprt/assert.h>
63#include <iprt/initterm.h>
64#include <iprt/process.h>
65#include <iprt/mem.h>
66#include <iprt/memobj.h>
67#include <iprt/asm.h>
68#include <iprt/timer.h>
69#include <iprt/heap.h>
70
71#define MODULE_NAME VBOXGUEST_MODULE_NAME
72
73/*
74 * IRQ related functions.
75 */
76static void VBoxGuestHaikuRemoveIRQ(void *pvState);
77static int VBoxGuestHaikuAddIRQ(void *pvState);
78static int32 VBoxGuestHaikuISR(void *pvState);
79
80/*
81 * Available functions for kernel drivers.
82 */
83DECLVBGL(int) VBoxGuestHaikuServiceCall(void *pvSession, unsigned uCmd, void *pvData, size_t cbData, size_t *pcbDataReturned);
84DECLVBGL(void *) VBoxGuestHaikuServiceOpen(uint32_t *pu32Version);
85DECLVBGL(int) VBoxGuestHaikuServiceClose(void *pvSession);
86DECLVBGL(void *) VBoxGuestIDCOpen(uint32_t *pu32Version);
87DECLVBGL(int) VBoxGuestIDCClose(void *pvSession);
88DECLVBGL(int) VBoxGuestIDCCall(void *pvSession, unsigned iCmd, void *pvData, size_t cbData, size_t *pcbDataReturned);
89
90static status_t std_ops(int32 op, ...);
91
92static RTSPINLOCK g_Spinlock = NIL_RTSPINLOCK;
93
94int32 api_version = B_CUR_DRIVER_API_VERSION;
95
96/** List of cloned device. Managed by the kernel. */
97//static struct clonedevs *g_pVBoxGuestHaikuClones;
98/** The dev_clone event handler tag. */
99//static eventhandler_tag g_VBoxGuestHaikuEHTag;
100/** selinfo structure used for polling. */
101//static struct selinfo g_SelInfo;
102/** PCI Bus Manager Module */
103static pci_module_info *gPCI;
104
105static struct vboxguest_module_info g_VBoxGuest =
106{
107 {
108 MODULE_NAME,
109 0,
110 std_ops
111 },
112 { 0 },
113 { 0 },
114 0,
115 RTLogBackdoorPrintf,
116 RTLogBackdoorPrintfV,
117 RTLogSetDefaultInstanceThread,
118 RTMemAllocExTag,
119 RTMemContAlloc,
120 RTMemContFree,
121 RTMemFreeEx,
122 RTMpIsCpuPossible,
123 RTMpNotificationDeregister,
124 RTMpNotificationRegister,
125 RTMpOnAll,
126 RTMpOnOthers,
127 RTMpOnSpecific,
128 RTPowerNotificationDeregister,
129 RTPowerNotificationRegister,
130 RTPowerSignalEvent,
131 RTR0AssertPanicSystem,
132 RTR0Init,
133 RTR0MemObjAddress,
134 RTR0MemObjAddressR3,
135 RTR0MemObjAllocContTag,
136 RTR0MemObjAllocLowTag,
137 RTR0MemObjAllocPageTag,
138 RTR0MemObjAllocPhysExTag,
139 RTR0MemObjAllocPhysNCTag,
140 RTR0MemObjAllocPhysTag,
141 RTR0MemObjEnterPhysTag,
142 RTR0MemObjFree,
143 RTR0MemObjGetPagePhysAddr,
144 RTR0MemObjIsMapping,
145 RTR0MemObjLockKernelTag,
146 RTR0MemObjLockUserTag,
147 RTR0MemObjMapKernelExTag,
148 RTR0MemObjMapKernelTag,
149 RTR0MemObjMapUserTag,
150 RTR0MemObjProtect,
151 RTR0MemObjReserveKernelTag,
152 RTR0MemObjReserveUserTag,
153 RTR0MemObjSize,
154 RTR0ProcHandleSelf,
155 RTR0Term,
156 RTR0TermForced,
157 RTProcSelf,
158 RTSemEventGetResolution,
159 RTSemEventMultiGetResolution,
160 RTSemEventMultiWaitEx,
161 RTSemEventMultiWaitExDebug,
162 RTSemEventWaitEx,
163 RTSemEventWaitExDebug,
164 RTThreadIsInInterrupt,
165 RTThreadPreemptDisable,
166 RTThreadPreemptIsEnabled,
167 RTThreadPreemptIsPending,
168 RTThreadPreemptIsPendingTrusty,
169 RTThreadPreemptIsPossible,
170 RTThreadPreemptRestore,
171 RTTimerGetSystemGranularity,
172 RTTimerReleaseSystemGranularity,
173 RTTimerRequestSystemGranularity,
174 RTSpinlockAcquire,
175 RTSpinlockRelease,
176 RTSpinlockReleaseNoInts,
177 RTMemTmpAllocTag,
178 RTMemTmpFree,
179 RTLogDefaultInstance,
180 RTLogRelDefaultInstance,
181 RTErrConvertToErrno,
182 VBoxGuestCommonIOCtl,
183 VBoxGuestCreateUserSession,
184 VBoxGuestCloseSession,
185 VBoxGuestIDCOpen,
186 VBoxGuestIDCClose,
187 VBoxGuestIDCCall,
188 RTAssertMsg1Weak,
189 RTAssertMsg2Weak,
190 RTAssertMsg2WeakV,
191 RTAssertShouldPanic,
192 RTSemFastMutexCreate,
193 RTSemFastMutexDestroy,
194 RTSemFastMutexRelease,
195 RTSemFastMutexRequest,
196 RTSemMutexCreate,
197 RTSemMutexDestroy,
198 RTSemMutexRelease,
199 RTSemMutexRequest,
200 RTHeapSimpleRelocate,
201 RTHeapOffsetInit,
202 RTHeapSimpleInit,
203 RTHeapOffsetAlloc,
204 RTHeapSimpleAlloc,
205 RTHeapOffsetFree,
206 RTHeapSimpleFree
207};
208
209#if 0
210/**
211 * DEVFS event handler.
212 */
213static void VBoxGuestHaikuClone(void *pvArg, struct ucred *pCred, char *pszName, int cchName, struct cdev **ppDev)
214{
215 int iUnit;
216 int rc;
217
218 Log(("VBoxGuestHaikuClone: pszName=%s ppDev=%p\n", pszName, ppDev));
219
220 /*
221 * One device node per user, si_drv1 points to the session.
222 * /dev/vboxguest<N> where N = {0...255}.
223 */
224 if (!ppDev)
225 return;
226 if (strcmp(pszName, "vboxguest") == 0)
227 iUnit = -1;
228 else if (dev_stdclone(pszName, NULL, "vboxguest", &iUnit) != 1)
229 return;
230 if (iUnit >= 256)
231 {
232 Log(("VBoxGuestHaikuClone: iUnit=%d >= 256 - rejected\n", iUnit));
233 return;
234 }
235
236 Log(("VBoxGuestHaikuClone: pszName=%s iUnit=%d\n", pszName, iUnit));
237
238 rc = clone_create(&g_pVBoxGuestHaikuClones, &g_VBoxGuestHaikuDeviceHooks, &iUnit, ppDev, 0);
239 Log(("VBoxGuestHaikuClone: clone_create -> %d; iUnit=%d\n", rc, iUnit));
240 if (rc)
241 {
242 *ppDev = make_dev(&g_VBoxGuestHaikuDeviceHooks,
243 iUnit,
244 UID_ROOT,
245 GID_WHEEL,
246 0644,
247 "vboxguest%d", iUnit);
248 if (*ppDev)
249 {
250 dev_ref(*ppDev);
251 (*ppDev)->si_flags |= SI_CHEAPCLONE;
252 Log(("VBoxGuestHaikuClone: Created *ppDev=%p iUnit=%d si_drv1=%p si_drv2=%p\n",
253 *ppDev, iUnit, (*ppDev)->si_drv1, (*ppDev)->si_drv2));
254 (*ppDev)->si_drv1 = (*ppDev)->si_drv2 = NULL;
255 }
256 else
257 Log(("VBoxGuestHaikuClone: make_dev iUnit=%d failed\n", iUnit));
258 }
259 else
260 Log(("VBoxGuestHaikuClone: Existing *ppDev=%p iUnit=%d si_drv1=%p si_drv2=%p\n",
261 *ppDev, iUnit, (*ppDev)->si_drv1, (*ppDev)->si_drv2));
262}
263#endif
264
265
266static status_t VBoxGuestHaikuDetach(void)
267{
268 struct VBoxGuestDeviceState *pState = &sState;
269
270 if (cUsers > 0)
271 return EBUSY;
272
273 /*
274 * Reverse what we did in VBoxGuestHaikuAttach.
275 */
276 VBoxGuestHaikuRemoveIRQ(pState);
277
278 if (pState->iVMMDevMemAreaId)
279 delete_area(pState->iVMMDevMemAreaId);
280
281 VBoxGuestDeleteDevExt(&g_DevExt);
282
283#ifdef DO_LOG
284 RTLogDestroy(RTLogRelSetDefaultInstance(NULL));
285 RTLogSetDefaultInstance(NULL);
286// RTLogDestroy(RTLogSetDefaultInstance(NULL));
287#endif
288
289 RTSpinlockDestroy(g_Spinlock);
290 g_Spinlock = NIL_RTSPINLOCK;
291
292 RTR0Term();
293 return B_OK;
294}
295
296
297/**
298 * Interrupt service routine.
299 *
300 * @returns Whether the interrupt was from VMMDev.
301 * @param pvState Opaque pointer to the device state.
302 */
303static int32 VBoxGuestHaikuISR(void *pvState)
304{
305 LogFlow((MODULE_NAME ":VBoxGuestHaikuISR pvState=%p\n", pvState));
306
307 bool fOurIRQ = VBoxGuestCommonISR(&g_DevExt);
308 if (fOurIRQ)
309 return B_HANDLED_INTERRUPT;
310 return B_UNHANDLED_INTERRUPT;
311}
312
313
314void VBoxGuestNativeISRMousePollEvent(PVBOXGUESTDEVEXT pDevExt)
315{
316 LogFlow((MODULE_NAME "::NativeISRMousePollEvent:\n"));
317
318 status_t err = B_OK;
319 //dprintf(MODULE_NAME ": isr mouse\n");
320
321 /*
322 * Wake up poll waiters.
323 */
324 //selwakeup(&g_SelInfo);
325 //XXX:notify_select_event();
326 RTSpinlockAcquire(g_Spinlock);
327
328 if (sState.selectSync)
329 {
330 //dprintf(MODULE_NAME ": isr mouse: notify\n");
331 notify_select_event(sState.selectSync, sState.selectEvent);
332 sState.selectEvent = (uint8_t)0;
333 sState.selectRef = (uint32_t)0;
334 sState.selectSync = NULL;
335 }
336 else
337 err = B_ERROR;
338
339 RTSpinlockRelease(g_Spinlock);
340}
341
342
343/**
344 * Sets IRQ for VMMDev.
345 *
346 * @returns Haiku error code.
347 * @param pvState Pointer to the state info structure.
348 */
349static int VBoxGuestHaikuAddIRQ(void *pvState)
350{
351 status_t err;
352 struct VBoxGuestDeviceState *pState = (struct VBoxGuestDeviceState *)pvState;
353
354 AssertReturn(pState, VERR_INVALID_PARAMETER);
355
356 err = install_io_interrupt_handler(pState->iIrqResId, VBoxGuestHaikuISR, pState, 0);
357 if (err == B_OK)
358 return VINF_SUCCESS;
359 return VERR_DEV_IO_ERROR;
360}
361
362
363/**
364 * Removes IRQ for VMMDev.
365 *
366 * @param pvState Opaque pointer to the state info structure.
367 */
368static void VBoxGuestHaikuRemoveIRQ(void *pvState)
369{
370 struct VBoxGuestDeviceState *pState = (struct VBoxGuestDeviceState *)pvState;
371 AssertPtr(pState);
372
373 remove_io_interrupt_handler(pState->iIrqResId, VBoxGuestHaikuISR, pState);
374}
375
376
377static status_t VBoxGuestHaikuAttach(const pci_info *pDevice)
378{
379 status_t status;
380 int rc = VINF_SUCCESS;
381 int iResId = 0;
382 struct VBoxGuestDeviceState *pState = &sState;
383 static const char *const s_apszGroups[] = VBOX_LOGGROUP_NAMES;
384 PRTLOGGER pRelLogger;
385
386 AssertReturn(pDevice, B_BAD_VALUE);
387
388 cUsers = 0;
389
390 /*
391 * Initialize IPRT R0 driver, which internally calls OS-specific r0 init.
392 */
393 rc = RTR0Init(0);
394 if (RT_FAILURE(rc))
395 {
396 /** @todo r=ramshankar: use dprintf here. */
397 LogFunc(("RTR0Init failed.\n"));
398 return ENXIO;
399 }
400
401 rc = RTSpinlockCreate(&g_Spinlock, RTSPINLOCK_FLAGS_INTERRUPT_SAFE, "VBoxGuestHaiku");
402 if (RT_FAILURE(rc))
403 {
404 LogRel(("VBoxGuestHaikuAttach: RTSpinlock create failed. rc=%Rrc\n", rc));
405 return ENXIO;
406 }
407
408#ifdef DO_LOG
409 /*
410 * Create the release log.
411 * (We do that here instead of common code because we want to log
412 * early failures using the LogRel macro.)
413 */
414 rc = RTLogCreate(&pRelLogger, 0 | RTLOGFLAGS_PREFIX_THREAD /* fFlags */, "all",
415 "VBOX_RELEASE_LOG", RT_ELEMENTS(s_apszGroups), s_apszGroups,
416 RTLOGDEST_STDOUT | RTLOGDEST_DEBUGGER | RTLOGDEST_USER, NULL);
417 dprintf(MODULE_NAME ": RTLogCreate: %d\n", rc);
418 if (RT_SUCCESS(rc))
419 {
420 //RTLogGroupSettings(pRelLogger, g_szLogGrp);
421 //RTLogFlags(pRelLogger, g_szLogFlags);
422 //RTLogDestinations(pRelLogger, "/var/log/vboxguest.log");
423 RTLogRelSetDefaultInstance(pRelLogger);
424 RTLogSetDefaultInstance(pRelLogger); //XXX
425 }
426#endif
427
428 /*
429 * Allocate I/O port resource.
430 */
431 pState->uIOPortBase = pDevice->u.h0.base_registers[0];
432 /* @todo check flags for IO? */
433 if (pState->uIOPortBase)
434 {
435 /*
436 * Map the MMIO region.
437 */
438 uint32 phys = pDevice->u.h0.base_registers[1];
439 /* @todo Check flags for mem? */
440 pState->VMMDevMemSize = pDevice->u.h0.base_register_sizes[1];
441 pState->iVMMDevMemAreaId = map_physical_memory("VirtualBox Guest MMIO", phys, pState->VMMDevMemSize,
442 B_ANY_KERNEL_BLOCK_ADDRESS, B_KERNEL_READ_AREA | B_KERNEL_WRITE_AREA,
443 &pState->pMMIOBase);
444 if (pState->iVMMDevMemAreaId > 0 && pState->pMMIOBase)
445 {
446 /*
447 * Call the common device extension initializer.
448 */
449 rc = VBoxGuestInitDevExt(&g_DevExt, pState->uIOPortBase, pState->pMMIOBase, pState->VMMDevMemSize,
450#if ARCH_BITS == 64
451 VBOXOSTYPE_Haiku_x64,
452#else
453 VBOXOSTYPE_Haiku,
454#endif
455 VMMDEV_EVENT_MOUSE_POSITION_CHANGED);
456 if (RT_SUCCESS(rc))
457 {
458 /*
459 * Add IRQ of VMMDev.
460 */
461 pState->iIrqResId = pDevice->u.h0.interrupt_line;
462 rc = VBoxGuestHaikuAddIRQ(pState);
463 if (RT_SUCCESS(rc))
464 {
465 LogRel((MODULE_NAME ": loaded successfully\n"));
466 return B_OK;
467 }
468
469 LogRel((MODULE_NAME ":VBoxGuestInitDevExt failed.\n"));
470 VBoxGuestDeleteDevExt(&g_DevExt);
471 }
472 else
473 LogRel((MODULE_NAME ":VBoxGuestHaikuAddIRQ failed.\n"));
474 }
475 else
476 LogRel((MODULE_NAME ":MMIO region setup failed.\n"));
477 }
478 else
479 LogRel((MODULE_NAME ":IOport setup failed.\n"));
480
481 RTR0Term();
482 return ENXIO;
483}
484
485
486static status_t VBoxGuestHaikuProbe(pci_info *pDevice)
487{
488 if ((pDevice->vendor_id == VMMDEV_VENDORID) && (pDevice->device_id == VMMDEV_DEVICEID))
489 return B_OK;
490
491 return ENXIO;
492}
493
494
495status_t init_module(void)
496{
497 status_t err = B_ENTRY_NOT_FOUND;
498 pci_info info;
499 int ix = 0;
500
501 err = get_module(B_PCI_MODULE_NAME, (module_info **)&gPCI);
502 if (err != B_OK)
503 return err;
504
505 while ((*gPCI->get_nth_pci_info)(ix++, &info) == B_OK)
506 {
507 if (VBoxGuestHaikuProbe(&info) == 0)
508 {
509 /* We found it */
510 err = VBoxGuestHaikuAttach(&info);
511 return err;
512 }
513 }
514
515 return B_ENTRY_NOT_FOUND;
516}
517
518
519void uninit_module(void)
520{
521 VBoxGuestHaikuDetach();
522 put_module(B_PCI_MODULE_NAME);
523}
524
525
526static status_t std_ops(int32 op, ...)
527{
528 switch (op)
529 {
530 case B_MODULE_INIT:
531 return init_module();
532
533 case B_MODULE_UNINIT:
534 {
535 uninit_module();
536 return B_OK;
537 }
538
539 default:
540 return B_ERROR;
541 }
542}
543
544
545_EXPORT module_info *modules[] =
546{
547 (module_info *)&g_VBoxGuest,
548 NULL
549};
550
551/* Common code that depend on g_DevExt. */
552#include "VBoxGuestIDC-unix.c.h"
553
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