VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/VBoxNetAdp/darwin/VBoxNetAdp-darwin.cpp@ 23312

Last change on this file since 23312 was 23082, checked in by vboxsync, 15 years ago

VBoxNetAdp: Darwin: fix missing vboxNetAdpFindByName, use common code, drop useless code.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 15.9 KB
Line 
1/* $Id: VBoxNetAdp-darwin.cpp 23082 2009-09-17 11:00:13Z vboxsync $ */
2/** @file
3 * VBoxNetAdp - Virtual Network Adapter Driver (Host), Darwin Specific Code.
4 */
5
6/*
7 * Copyright (C) 2008 Sun Microsystems, Inc.
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 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa
18 * Clara, CA 95054 USA or visit http://www.sun.com if you need
19 * additional information or have any questions.
20 */
21
22/*******************************************************************************
23* Header Files *
24*******************************************************************************/
25/*
26 * Deal with conflicts first.
27 * PVM - BSD mess, that FreeBSD has correct a long time ago.
28 * iprt/types.h before sys/param.h - prevents UINT32_C and friends.
29 */
30#include <iprt/types.h>
31#include <sys/param.h>
32#undef PVM
33
34#define LOG_GROUP LOG_GROUP_NET_ADP_DRV
35#include <VBox/log.h>
36#include <VBox/err.h>
37#include <VBox/version.h>
38#include <iprt/assert.h>
39#include <iprt/initterm.h>
40#include <iprt/semaphore.h>
41#include <iprt/spinlock.h>
42#include <iprt/string.h>
43#include <iprt/uuid.h>
44#include <iprt/alloca.h>
45
46#include <sys/systm.h>
47RT_C_DECLS_BEGIN /* Buggy 10.4 headers, fixed in 10.5. */
48#include <sys/kpi_mbuf.h>
49RT_C_DECLS_END
50
51#include <net/ethernet.h>
52#include <net/if_ether.h>
53#include <net/if_types.h>
54#include <sys/socket.h>
55#include <net/if.h>
56#include <net/if_dl.h>
57#include <sys/errno.h>
58#include <sys/param.h>
59#include <sys/conf.h>
60#include <miscfs/devfs/devfs.h>
61
62#define VBOXNETADP_OS_SPECFIC 1
63#include "../VBoxNetAdpInternal.h"
64
65/*******************************************************************************
66* Defined Constants And Macros *
67*******************************************************************************/
68/** The maximum number of SG segments.
69 * Used to prevent stack overflow and similar bad stuff. */
70#define VBOXNETADP_DARWIN_MAX_SEGS 32
71#define VBOXNETADP_DARWIN_MAX_FAMILIES 4
72#define VBOXNETADP_DARWIN_NAME "vboxnet"
73#define VBOXNETADP_DARWIN_MTU 1500
74#define VBOXNETADP_DARWIN_DETACH_TIMEOUT 500
75
76#define VBOXNETADP_FROM_IFACE(iface) ((PVBOXNETADP) ifnet_softc(iface))
77
78/* debug printf */
79#if defined(RT_OS_WINDOWS)
80# define OSDBGPRINT(a) DbgPrint a
81#elif defined(RT_OS_LINUX)
82# define OSDBGPRINT(a) printk a
83#elif defined(RT_OS_DARWIN)
84# define OSDBGPRINT(a) printf a
85#elif defined(RT_OS_OS2)
86# define OSDBGPRINT(a) SUPR0Printf a
87#elif defined(RT_OS_FREEBSD)
88# define OSDBGPRINT(a) printf a
89#elif defined(RT_OS_SOLARIS)
90# define OSDBGPRINT(a) SUPR0Printf a
91#else
92# define OSDBGPRINT(a)
93#endif
94
95/*******************************************************************************
96* Internal Functions *
97*******************************************************************************/
98RT_C_DECLS_BEGIN
99static kern_return_t VBoxNetAdpDarwinStart(struct kmod_info *pKModInfo, void *pvData);
100static kern_return_t VBoxNetAdpDarwinStop(struct kmod_info *pKModInfo, void *pvData);
101RT_C_DECLS_END
102
103static int VBoxNetAdpDarwinOpen(dev_t Dev, int fFlags, int fDevType, struct proc *pProcess);
104static int VBoxNetAdpDarwinClose(dev_t Dev, int fFlags, int fDevType, struct proc *pProcess);
105static int VBoxNetAdpDarwinIOCtl(dev_t Dev, u_long iCmd, caddr_t pData, int fFlags, struct proc *pProcess);
106
107/*******************************************************************************
108* Global Variables *
109*******************************************************************************/
110/**
111 * Declare the module stuff.
112 */
113RT_C_DECLS_BEGIN
114extern kern_return_t _start(struct kmod_info *pKModInfo, void *pvData);
115extern kern_return_t _stop(struct kmod_info *pKModInfo, void *pvData);
116
117KMOD_EXPLICIT_DECL(VBoxNetAdp, VBOX_VERSION_STRING, _start, _stop)
118DECLHIDDEN(kmod_start_func_t *) _realmain = VBoxNetAdpDarwinStart;
119DECLHIDDEN(kmod_stop_func_t *) _antimain = VBoxNetAdpDarwinStop;
120DECLHIDDEN(int) _kext_apple_cc = __APPLE_CC__;
121RT_C_DECLS_END
122
123/**
124 * The (common) global data.
125 */
126static int g_nCtlDev = -1; /* Major dev number */
127static void *g_hCtlDev = 0; /* FS dev handle */
128
129/**
130 * The character device switch table for the driver.
131 */
132static struct cdevsw g_ChDev =
133{
134 /*.d_open = */VBoxNetAdpDarwinOpen,
135 /*.d_close = */VBoxNetAdpDarwinClose,
136 /*.d_read = */eno_rdwrt,
137 /*.d_write = */eno_rdwrt,
138 /*.d_ioctl = */VBoxNetAdpDarwinIOCtl,
139 /*.d_stop = */eno_stop,
140 /*.d_reset = */eno_reset,
141 /*.d_ttys = */NULL,
142 /*.d_select = */eno_select,
143 /*.d_mmap = */eno_mmap,
144 /*.d_strategy = */eno_strat,
145 /*.d_getc = */eno_getc,
146 /*.d_putc = */eno_putc,
147 /*.d_type = */0
148};
149
150
151
152static void vboxNetAdpDarwinComposeUUID(PVBOXNETADP pThis, PRTUUID pUuid)
153{
154 /* Generate UUID from name and MAC address. */
155 RTUuidClear(pUuid);
156 memcpy(pUuid->au8, "vboxnet", 7);
157 pUuid->Gen.u8ClockSeqHiAndReserved = (pUuid->Gen.u8ClockSeqHiAndReserved & 0x3f) | 0x80;
158 pUuid->Gen.u16TimeHiAndVersion = (pUuid->Gen.u16TimeHiAndVersion & 0x0fff) | 0x4000;
159 pUuid->Gen.u8ClockSeqLow = pThis->uUnit;
160 vboxNetAdpComposeMACAddress(pThis, (PRTMAC)pUuid->Gen.au8Node);
161}
162
163static errno_t vboxNetAdpDarwinOutput(ifnet_t pIface, mbuf_t pMBuf)
164{
165 mbuf_freem_list(pMBuf);
166 return 0;
167}
168
169static void vboxNetAdpDarwinAttachFamily(PVBOXNETADP pThis, protocol_family_t Family)
170{
171 u_int32_t i;
172 for (i = 0; i < VBOXNETADP_MAX_FAMILIES; i++)
173 if (pThis->u.s.aAttachedFamilies[i] == 0)
174 {
175 pThis->u.s.aAttachedFamilies[i] = Family;
176 break;
177 }
178}
179
180static void vboxNetAdpDarwinDetachFamily(PVBOXNETADP pThis, protocol_family_t Family)
181{
182 u_int32_t i;
183 for (i = 0; i < VBOXNETADP_MAX_FAMILIES; i++)
184 if (pThis->u.s.aAttachedFamilies[i] == Family)
185 pThis->u.s.aAttachedFamilies[i] = 0;
186}
187
188static errno_t vboxNetAdpDarwinAddProto(ifnet_t pIface, protocol_family_t Family, const struct ifnet_demux_desc *pDemuxDesc, u_int32_t nDesc)
189{
190 PVBOXNETADP pThis = VBOXNETADP_FROM_IFACE(pIface);
191 Assert(pThis);
192 vboxNetAdpDarwinAttachFamily(pThis, Family);
193 LogFlow(("vboxNetAdpAddProto: Family=%d.\n", Family));
194 return ether_add_proto(pIface, Family, pDemuxDesc, nDesc);
195}
196
197static errno_t vboxNetAdpDarwinDelProto(ifnet_t pIface, protocol_family_t Family)
198{
199 PVBOXNETADP pThis = VBOXNETADP_FROM_IFACE(pIface);
200 Assert(pThis);
201 LogFlow(("vboxNetAdpDelProto: Family=%d.\n", Family));
202 vboxNetAdpDarwinDetachFamily(pThis, Family);
203 return ether_del_proto(pIface, Family);
204}
205
206static void vboxNetAdpDarwinDetach(ifnet_t pIface)
207{
208 PVBOXNETADP pThis = VBOXNETADP_FROM_IFACE(pIface);
209 Assert(pThis);
210 Log2(("vboxNetAdpDarwinDetach: Signaling detach to vboxNetAdpUnregisterDevice.\n"));
211 /* Let vboxNetAdpDarwinUnregisterDevice know that the interface has been detached. */
212 RTSemEventSignal(pThis->u.s.hEvtDetached);
213}
214
215
216int vboxNetAdpOsCreate(PVBOXNETADP pThis, PCRTMAC pMACAddress)
217{
218 int rc;
219 struct ifnet_init_params Params;
220 RTUUID uuid;
221 struct sockaddr_dl mac;
222
223 pThis->u.s.hEvtDetached = NIL_RTSEMEVENT;
224 rc = RTSemEventCreate(&pThis->u.s.hEvtDetached);
225 if (RT_FAILURE(rc))
226 {
227 OSDBGPRINT(("vboxNetAdpOsCreate: failed to create semaphore (rc=%d).\n", rc));
228 return rc;
229 }
230
231 mac.sdl_len = sizeof(mac);
232 mac.sdl_family = AF_LINK;
233 mac.sdl_alen = ETHER_ADDR_LEN;
234 mac.sdl_nlen = 0;
235 mac.sdl_slen = 0;
236 memcpy(LLADDR(&mac), pMACAddress->au8, mac.sdl_alen);
237
238 RTStrPrintf(pThis->szName, VBOXNETADP_MAX_NAME_LEN, "%s%d", VBOXNETADP_NAME, pThis->uUnit);
239 vboxNetAdpDarwinComposeUUID(pThis, &uuid);
240 Params.uniqueid = uuid.au8;
241 Params.uniqueid_len = sizeof(uuid);
242 Params.name = VBOXNETADP_NAME;
243 Params.unit = pThis->uUnit;
244 Params.family = IFNET_FAMILY_ETHERNET;
245 Params.type = IFT_ETHER;
246 Params.output = vboxNetAdpDarwinOutput;
247 Params.demux = ether_demux;
248 Params.add_proto = vboxNetAdpDarwinAddProto;
249 Params.del_proto = vboxNetAdpDarwinDelProto;
250 Params.check_multi = ether_check_multi;
251 Params.framer = ether_frameout;
252 Params.softc = pThis;
253 Params.ioctl = (ifnet_ioctl_func)ether_ioctl;
254 Params.set_bpf_tap = NULL;
255 Params.detach = vboxNetAdpDarwinDetach;
256 Params.event = NULL;
257 Params.broadcast_addr = "\xFF\xFF\xFF\xFF\xFF\xFF";
258 Params.broadcast_len = ETHER_ADDR_LEN;
259
260 errno_t err = ifnet_allocate(&Params, &pThis->u.s.pIface);
261 if (!err)
262 {
263 err = ifnet_attach(pThis->u.s.pIface, &mac);
264 if (!err)
265 {
266 err = ifnet_set_flags(pThis->u.s.pIface, IFF_RUNNING | IFF_BROADCAST | IFF_SIMPLEX | IFF_MULTICAST, 0xFFFF);
267 if (!err)
268 {
269 ifnet_set_mtu(pThis->u.s.pIface, VBOXNETADP_MTU);
270 return VINF_SUCCESS;
271 }
272 else
273 Log(("vboxNetAdpDarwinRegisterDevice: Failed to set flags (err=%d).\n", err));
274 ifnet_detach(pThis->u.s.pIface);
275 }
276 else
277 Log(("vboxNetAdpDarwinRegisterDevice: Failed to attach to interface (err=%d).\n", err));
278 ifnet_release(pThis->u.s.pIface);
279 }
280 else
281 Log(("vboxNetAdpDarwinRegisterDevice: Failed to allocate interface (err=%d).\n", err));
282
283 RTSemEventDestroy(pThis->u.s.hEvtDetached);
284 pThis->u.s.hEvtDetached = NIL_RTSEMEVENT;
285
286 return RTErrConvertFromErrno(err);
287}
288
289void vboxNetAdpOsDestroy(PVBOXNETADP pThis)
290{
291 u_int32_t i;
292 /* Bring down the interface */
293 int rc = VINF_SUCCESS;
294 errno_t err;
295
296 AssertPtr(pThis->u.s.pIface);
297 Assert(pThis->u.s.hEvtDetached != NIL_RTSEMEVENT);
298
299 err = ifnet_set_flags(pThis->u.s.pIface, 0, IFF_UP | IFF_RUNNING);
300 if (err)
301 Log(("vboxNetAdpDarwinUnregisterDevice: Failed to bring down interface "
302 "(err=%d).\n", err));
303 /* Detach all protocols. */
304 for (i = 0; i < VBOXNETADP_MAX_FAMILIES; i++)
305 if (pThis->u.s.aAttachedFamilies[i])
306 ifnet_detach_protocol(pThis->u.s.pIface, pThis->u.s.aAttachedFamilies[i]);
307 err = ifnet_detach(pThis->u.s.pIface);
308 if (err)
309 Log(("vboxNetAdpDarwinUnregisterDevice: Failed to detach interface "
310 "(err=%d).\n", err));
311 Log2(("vboxNetAdpDarwinUnregisterDevice: Waiting for 'detached' event...\n"));
312 /* Wait until we get a signal from detach callback. */
313 rc = RTSemEventWait(pThis->u.s.hEvtDetached, VBOXNETADP_DETACH_TIMEOUT);
314 if (rc == VERR_TIMEOUT)
315 LogRel(("VBoxAdpDrv: Failed to detach interface %s%d\n.",
316 VBOXNETADP_NAME, pThis->uUnit));
317 err = ifnet_release(pThis->u.s.pIface);
318 if (err)
319 Log(("vboxNetAdpUnregisterDevice: Failed to release interface (err=%d).\n", err));
320
321 RTSemEventDestroy(pThis->u.s.hEvtDetached);
322 pThis->u.s.hEvtDetached = NIL_RTSEMEVENT;
323}
324
325/**
326 * Device open. Called on open /dev/vboxnetctl
327 *
328 * @param pInode Pointer to inode info structure.
329 * @param pFilp Associated file pointer.
330 */
331static int VBoxNetAdpDarwinOpen(dev_t Dev, int fFlags, int fDevType, struct proc *pProcess)
332{
333 char szName[128];
334 szName[0] = '\0';
335 proc_name(proc_pid(pProcess), szName, sizeof(szName));
336 Log(("VBoxNetAdpDarwinOpen: pid=%d '%s'\n", proc_pid(pProcess), szName));
337 return 0;
338}
339
340/**
341 * Close device.
342 */
343static int VBoxNetAdpDarwinClose(dev_t Dev, int fFlags, int fDevType, struct proc *pProcess)
344{
345 Log(("VBoxNetAdpDarwinClose: pid=%d\n", proc_pid(pProcess)));
346 return 0;
347}
348
349/**
350 * Device I/O Control entry point.
351 *
352 * @returns Darwin for slow IOCtls and VBox status code for the fast ones.
353 * @param Dev The device number (major+minor).
354 * @param iCmd The IOCtl command.
355 * @param pData Pointer to the data (if any it's a SUPDRVIOCTLDATA (kernel copy)).
356 * @param fFlags Flag saying we're a character device (like we didn't know already).
357 * @param pProcess The process issuing this request.
358 */
359static int VBoxNetAdpDarwinIOCtl(dev_t Dev, u_long iCmd, caddr_t pData, int fFlags, struct proc *pProcess)
360{
361 uint32_t cbReq = IOCPARM_LEN(iCmd);
362 PVBOXNETADPREQ pReq = (PVBOXNETADPREQ)pData;
363 int rc;
364
365 Log(("VBoxNetAdpDarwinIOCtl: param len %#x; iCmd=%#lx\n", cbReq, iCmd));
366 switch (IOCBASECMD(iCmd))
367 {
368 case IOCBASECMD(VBOXNETADP_CTL_ADD):
369 {
370 if ( (IOC_DIRMASK & iCmd) != IOC_OUT
371 || cbReq < sizeof(VBOXNETADPREQ))
372 return EINVAL;
373
374 PVBOXNETADP pNew;
375 rc = vboxNetAdpCreate(&pNew);
376 if (RT_FAILURE(rc))
377 return EINVAL;
378
379 Assert(strlen(pReq->szName) < sizeof(pReq->szName));
380 strncpy(pReq->szName, pNew->szName, sizeof(pReq->szName) - 1);
381 pReq->szName[sizeof(pReq->szName) - 1] = '\0';
382 Log(("VBoxNetAdpDarwinIOCtl: Added '%s'\n", pReq->szName));
383 break;
384 }
385
386 case IOCBASECMD(VBOXNETADP_CTL_REMOVE):
387 {
388 if (!memchr(pReq->szName, '\0', RT_MIN(cbReq, sizeof(pReq->szName))))
389 return EINVAL;
390
391 PVBOXNETADP pAdp = vboxNetAdpFindByName(pReq->szName);
392 if (!pAdp)
393 return EINVAL;
394
395 rc = vboxNetAdpDestroy(pAdp);
396 if (RT_FAILURE(rc))
397 return EINVAL;
398 Log(("VBoxNetAdpDarwinIOCtl: Removed %s\n", pReq->szName));
399 break;
400 }
401
402 default:
403 OSDBGPRINT(("VBoxNetAdpDarwinIOCtl: unknown command %x.\n", IOCBASECMD(iCmd)));
404 return EINVAL;
405 }
406
407 return 0;
408}
409
410int vboxNetAdpOsInit(PVBOXNETADP pThis)
411{
412 /*
413 * Init the darwin specific members.
414 */
415 pThis->u.s.pIface = NULL;
416 pThis->u.s.hEvtDetached = NIL_RTSEMEVENT;
417 memset(pThis->u.s.aAttachedFamilies, 0, sizeof(pThis->u.s.aAttachedFamilies));
418
419 return VINF_SUCCESS;
420}
421
422/**
423 * Start the kernel module.
424 */
425static kern_return_t VBoxNetAdpDarwinStart(struct kmod_info *pKModInfo, void *pvData)
426{
427 int rc;
428
429 /*
430 * Initialize IPRT and find our module tag id.
431 * (IPRT is shared with VBoxDrv, it creates the loggers.)
432 */
433 rc = RTR0Init(0);
434 if (RT_SUCCESS(rc))
435 {
436 Log(("VBoxNetAdpDarwinStart\n"));
437 rc = vboxNetAdpInit();
438 if (RT_SUCCESS(rc))
439 {
440 g_nCtlDev = cdevsw_add(-1, &g_ChDev);
441 if (g_nCtlDev < 0)
442 {
443 LogRel(("VBoxAdp: failed to register control device."));
444 rc = VERR_CANT_CREATE;
445 }
446 else
447 {
448 g_hCtlDev = devfs_make_node(makedev(g_nCtlDev, 0), DEVFS_CHAR,
449 UID_ROOT, GID_WHEEL, 0600, VBOXNETADP_CTL_DEV_NAME);
450 if (!g_hCtlDev)
451 {
452 LogRel(("VBoxAdp: failed to create FS node for control device."));
453 rc = VERR_CANT_CREATE;
454 }
455 }
456 }
457
458 if (RT_SUCCESS(rc))
459 {
460 LogRel(("VBoxAdpDrv: version " VBOX_VERSION_STRING " r%d\n", VBOX_SVN_REV));
461 return KMOD_RETURN_SUCCESS;
462 }
463
464 LogRel(("VBoxAdpDrv: failed to initialize device extension (rc=%d)\n", rc));
465 RTR0Term();
466 }
467 else
468 printf("VBoxAdpDrv: failed to initialize IPRT (rc=%d)\n", rc);
469
470 return KMOD_RETURN_FAILURE;
471}
472
473
474/**
475 * Stop the kernel module.
476 */
477static kern_return_t VBoxNetAdpDarwinStop(struct kmod_info *pKModInfo, void *pvData)
478{
479 Log(("VBoxNetAdpDarwinStop\n"));
480
481 vboxNetAdpShutdown();
482 /* Remove control device */
483 devfs_remove(g_hCtlDev);
484 cdevsw_remove(g_nCtlDev, &g_ChDev);
485
486 RTR0Term();
487
488 return KMOD_RETURN_SUCCESS;
489}
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