/** @file * SUPLib - FreeBSD Hosts, */ /* * Copyright (C) 2006-2007 innotek GmbH * * This file is part of VirtualBox Open Source Edition (OSE), as * available from http://www.virtualbox.org. This file is free software; * you can redistribute it and/or modify it under the terms of the GNU * General Public License (GPL) as published by the Free Software * Foundation, in version 2 as it comes in the "COPYING" file of the * VirtualBox OSE distribution. VirtualBox OSE is distributed in the * hope that it will be useful, but WITHOUT ANY WARRANTY of any kind. * * The contents of this file may alternatively be used under the terms * of the Common Development and Distribution License Version 1.0 * (CDDL) only, as it comes in the "COPYING.CDDL" file of the * VirtualBox OSE distribution, in which case the provisions of the * CDDL are applicable instead of those of the GPL. * * You may elect to license modified versions of this file under the * terms and conditions of either the GPL or the CDDL or both. */ /******************************************************************************* * Header Files * *******************************************************************************/ #define LOG_GROUP LOG_GROUP_SUP #include #include #include #include #include #include #include #include #include #include #include "SUPLibInternal.h" #include "SUPDRVIOC.h" #include #include #include #include #include /******************************************************************************* * Defined Constants And Macros * *******************************************************************************/ /** FreeBSD base device name. */ #define DEVICE_NAME "/dev/vboxdrv" /******************************************************************************* * Global Variables * *******************************************************************************/ /** Handle to the open device. */ static int g_hDevice = -1; int suplibOsInit(size_t cbReserve) { /* * Check if already initialized. */ if (g_hDevice >= 0) return VINF_SUCCESS; /* * Try open the BSD device. */ char szDevice[sizeof(DEVICE_NAME) + 16]; for (unsigned iUnit = 0; iUnit < 1024; iUnit++) { errno = 0; RTStrPrintf(szDevice, sizeof(szDevice), DEVICE_NAME "%d", iUnit); g_hDevice = open(szDevice, O_RDWR, 0); if (g_hDevice >= 0 || errno != EBUSY) break; } if (g_hDevice < 0) { int rc; switch (errno) { case ENODEV: rc = VERR_VM_DRIVER_LOAD_ERROR; break; case EPERM: case EACCES: rc = VERR_VM_DRIVER_NOT_ACCESSIBLE; break; case ENOENT: rc = VERR_VM_DRIVER_NOT_INSTALLED; break; default: rc = VERR_VM_DRIVER_OPEN_ERROR; break; } LogRel(("Failed to open \"%s\", errno=%d, rc=%Vrc\n", szDevice, errno, rc)); return rc; } /* * Mark the file handle close on exec. */ if (fcntl(g_hDevice, F_SETFD, FD_CLOEXEC) != 0) { int rc = errno; LogRel(("suplibOSInit: setting FD_CLOEXEC failed, errno=%d\n", rc)); close(g_hDevice); g_hDevice = -1; return RTErrConvertFromErrno(rc); } /* * We're done. */ NOREF(cbReserve); return VINF_SUCCESS; } int suplibOsTerm(void) { /* * Check if we're initited at all. */ if (g_hDevice >= 0) { if (close(g_hDevice)) AssertFailed(); g_hDevice = -1; } return VINF_SUCCESS; } int suplibOsInstall(void) { return VERR_NOT_IMPLEMENTED; } int suplibOsUninstall(void) { return VERR_NOT_IMPLEMENTED; } int suplibOsIOCtl(uintptr_t uFunction, void *pvReq, size_t cbReq) { AssertMsg(g_hDevice != -1, ("SUPLIB not initiated successfully!\n")); if (RT_LIKELY(ioctl(g_hDevice, uFunction, pvReq) >= 0)) return VINF_SUCCESS; return RTErrConvertFromErrno(errno); } int suplibOsIOCtlFast(uintptr_t uFunction) { int rc = ioctl(g_hDevice, uFunction, NULL); if (rc == -1) rc = errno; return rc; } int suplibOsPageAlloc(size_t cPages, void **ppvPages) { *ppvPages = RTMemPageAllocZ(cPages << PAGE_SHIFT); if (*ppvPages) return VINF_SUCCESS; return RTErrConvertFromErrno(errno); } int suplibOsPageFree(void *pvPages, size_t /* cPages */) { RTMemPageFree(pvPages); return VINF_SUCCESS; }