VirtualBox

source: vbox/trunk/src/VBox/HostDrivers/adpctl/VBoxNetAdpCtl.cpp@ 66372

Last change on this file since 66372 was 65649, checked in by vboxsync, 8 years ago

gcc 7: VBoxNetAdpCtl: fall thru

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 14.9 KB
Line 
1/* $Id: VBoxNetAdpCtl.cpp 65649 2017-02-07 11:43:53Z vboxsync $ */
2/** @file
3 * Apps - VBoxAdpCtl, Configuration tool for vboxnetX adapters.
4 */
5
6/*
7 * Copyright (C) 2009-2016 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
20/*********************************************************************************************************************************
21* Header Files *
22*********************************************************************************************************************************/
23#include <stdio.h>
24#include <stdlib.h>
25#include <string.h>
26#include <unistd.h>
27#include <sys/wait.h>
28#include <sys/ioctl.h>
29#include <sys/stat.h>
30#include <fcntl.h>
31#ifdef RT_OS_LINUX
32# include <net/if.h>
33# include <linux/types.h>
34/* Older versions of ethtool.h rely on these: */
35typedef unsigned long long u64;
36typedef __uint32_t u32;
37typedef __uint16_t u16;
38typedef __uint8_t u8;
39# include <limits.h> /* for INT_MAX */
40# include <linux/ethtool.h>
41# include <linux/sockios.h>
42#endif
43#ifdef RT_OS_SOLARIS
44# include <sys/ioccom.h>
45#endif
46
47/** @todo Error codes must be moved to some header file */
48#define ADPCTLERR_BAD_NAME 2
49#define ADPCTLERR_NO_CTL_DEV 3
50#define ADPCTLERR_IOCTL_FAILED 4
51#define ADPCTLERR_SOCKET_FAILED 5
52
53/** @todo These are duplicates from src/VBox/HostDrivers/VBoxNetAdp/VBoxNetAdpInternal.h */
54#define VBOXNETADP_CTL_DEV_NAME "/dev/vboxnetctl"
55#define VBOXNETADP_MAX_INSTANCES 128
56#define VBOXNETADP_NAME "vboxnet"
57#define VBOXNETADP_MAX_NAME_LEN 32
58#define VBOXNETADP_CTL_ADD _IOWR('v', 1, VBOXNETADPREQ)
59#define VBOXNETADP_CTL_REMOVE _IOW('v', 2, VBOXNETADPREQ)
60typedef struct VBoxNetAdpReq
61{
62 char szName[VBOXNETADP_MAX_NAME_LEN];
63} VBOXNETADPREQ;
64typedef VBOXNETADPREQ *PVBOXNETADPREQ;
65
66#define VBOXADPCTL_IFCONFIG_PATH1 "/sbin/ifconfig"
67#define VBOXADPCTL_IFCONFIG_PATH2 "/bin/ifconfig"
68static char *g_pszIfConfig;
69
70#if defined(RT_OS_LINUX)
71# define VBOXADPCTL_DEL_CMD "del"
72# define VBOXADPCTL_ADD_CMD "add"
73#elif defined(RT_OS_SOLARIS)
74# define VBOXADPCTL_DEL_CMD "removeif"
75# define VBOXADPCTL_ADD_CMD "addif"
76#else
77# define VBOXADPCTL_DEL_CMD "delete"
78# define VBOXADPCTL_ADD_CMD "add"
79#endif
80
81static void showUsage(void)
82{
83 fprintf(stderr, "Usage: VBoxNetAdpCtl <adapter> <address> ([netmask <address>] | remove)\n");
84 fprintf(stderr, " | VBoxNetAdpCtl [<adapter>] add\n");
85 fprintf(stderr, " | VBoxNetAdpCtl <adapter> remove\n");
86}
87
88static void setPathIfConfig(void)
89{
90 struct stat s;
91 if ( !stat(VBOXADPCTL_IFCONFIG_PATH1, &s)
92 && S_ISREG(s.st_mode))
93 g_pszIfConfig = (char*)VBOXADPCTL_IFCONFIG_PATH1;
94 else
95 g_pszIfConfig = (char*)VBOXADPCTL_IFCONFIG_PATH2;
96}
97
98static int executeIfconfig(const char *pcszAdapterName, const char *pcszArg1,
99 const char *pcszArg2 = NULL,
100 const char *pcszArg3 = NULL,
101 const char *pcszArg4 = NULL,
102 const char *pcszArg5 = NULL)
103{
104 const char * const argv[] =
105 {
106 g_pszIfConfig,
107 pcszAdapterName,
108 pcszArg1, /* [address family] */
109 pcszArg2, /* address */
110 pcszArg3, /* ['netmask'] */
111 pcszArg4, /* [network mask] */
112 pcszArg5, /* [network mask] */
113 NULL /* terminator */
114 };
115 char * const envp[] = { (char*)"LC_ALL=C", NULL };
116 int rc = EXIT_SUCCESS;
117 pid_t childPid = fork();
118 switch (childPid)
119 {
120 case -1: /* Something went wrong. */
121 perror("fork() failed");
122 rc = EXIT_FAILURE;
123 break;
124 case 0: /* Child process. */
125 if (execve(argv[0], (char * const*)argv, envp) == -1)
126 rc = EXIT_FAILURE;
127 break;
128 default: /* Parent process. */
129 waitpid(childPid, &rc, 0);
130 break;
131 }
132
133 return rc;
134}
135
136#define MAX_ADDRESSES 128
137#define MAX_ADDRLEN 64
138
139static bool removeAddresses(char *pszAdapterName)
140{
141 char szBuf[1024];
142 char aszAddresses[MAX_ADDRESSES][MAX_ADDRLEN];
143 int rc;
144 int fds[2];
145 char * const argv[] = { g_pszIfConfig, pszAdapterName, NULL };
146 char * const envp[] = { (char*)"LC_ALL=C", NULL };
147
148 memset(aszAddresses, 0, sizeof(aszAddresses));
149
150 rc = pipe(fds);
151 if (rc < 0)
152 return false;
153
154 pid_t pid = fork();
155 if (pid < 0)
156 return false;
157
158 if (pid == 0)
159 {
160 /* child */
161 close(fds[0]);
162 close(STDOUT_FILENO);
163 rc = dup2(fds[1], STDOUT_FILENO);
164 if (rc >= 0)
165 execve(argv[0], argv, envp);
166 return false;
167 }
168
169 /* parent */
170 close(fds[1]);
171 FILE *fp = fdopen(fds[0], "r");
172 if (!fp)
173 return false;
174
175 int cAddrs;
176 for (cAddrs = 0; cAddrs < MAX_ADDRESSES && fgets(szBuf, sizeof(szBuf), fp);)
177 {
178 int cbSkipWS = strspn(szBuf, " \t");
179 char *pszWord = strtok(szBuf + cbSkipWS, " ");
180 /* We are concerned with IPv6 address lines only. */
181 if (!pszWord || strcmp(pszWord, "inet6"))
182 continue;
183#ifdef RT_OS_LINUX
184 pszWord = strtok(NULL, " ");
185 /* Skip "addr:". */
186 if (!pszWord || strcmp(pszWord, "addr:"))
187 continue;
188#endif
189 pszWord = strtok(NULL, " ");
190 /* Skip link-local addresses. */
191 if (!pszWord || !strncmp(pszWord, "fe80", 4))
192 continue;
193 strncpy(aszAddresses[cAddrs++], pszWord, MAX_ADDRLEN-1);
194 }
195 fclose(fp);
196
197 for (int i = 0; i < cAddrs; i++)
198 {
199 if (executeIfconfig(pszAdapterName, "inet6",
200 VBOXADPCTL_DEL_CMD, aszAddresses[i]) != EXIT_SUCCESS)
201 return false;
202 }
203
204 return true;
205}
206
207
208#ifndef RT_OS_SOLARIS
209static int doIOCtl(unsigned long iCmd, VBOXNETADPREQ *pReq)
210{
211 int fd = open(VBOXNETADP_CTL_DEV_NAME, O_RDWR);
212 if (fd == -1)
213 {
214 fprintf(stderr, "VBoxNetAdpCtl: Error while %s %s: ",
215 iCmd == VBOXNETADP_CTL_REMOVE ? "removing" : "adding",
216 pReq->szName[0] ? pReq->szName : "new interface");
217 perror("failed to open " VBOXNETADP_CTL_DEV_NAME);
218 return ADPCTLERR_NO_CTL_DEV;
219 }
220
221 int rc = ioctl(fd, iCmd, pReq);
222 if (rc == -1)
223 {
224 fprintf(stderr, "VBoxNetAdpCtl: Error while %s %s: ",
225 iCmd == VBOXNETADP_CTL_REMOVE ? "removing" : "adding",
226 pReq->szName[0] ? pReq->szName : "new interface");
227 perror("VBoxNetAdpCtl: ioctl failed for " VBOXNETADP_CTL_DEV_NAME);
228 rc = ADPCTLERR_IOCTL_FAILED;
229 }
230
231 close(fd);
232
233 return rc;
234}
235#endif /* !RT_OS_SOLARIS */
236
237
238static int checkAdapterName(const char *pcszNameIn, char *pszNameOut)
239{
240 int iAdapterIndex = -1;
241
242 if ( strlen(pcszNameIn) >= VBOXNETADP_MAX_NAME_LEN
243 || sscanf(pcszNameIn, "vboxnet%d", &iAdapterIndex) != 1
244 || iAdapterIndex < 0 || iAdapterIndex >= VBOXNETADP_MAX_INSTANCES )
245 {
246 fprintf(stderr, "VBoxNetAdpCtl: Setting configuration for '%s' is not supported.\n", pcszNameIn);
247 return ADPCTLERR_BAD_NAME;
248 }
249 sprintf(pszNameOut, "vboxnet%d", iAdapterIndex);
250 if (strcmp(pszNameOut, pcszNameIn))
251 {
252 fprintf(stderr, "VBoxNetAdpCtl: Invalid adapter name '%s'.\n", pcszNameIn);
253 return ADPCTLERR_BAD_NAME;
254 }
255
256 return 0;
257}
258
259int main(int argc, char *argv[])
260{
261 char szAdapterName[VBOXNETADP_MAX_NAME_LEN];
262 char *pszAdapterName = NULL;
263 const char *pszAddress = NULL;
264 const char *pszNetworkMask = NULL;
265 const char *pszOption = NULL;
266 int rc = EXIT_SUCCESS;
267 bool fRemove = false;
268 VBOXNETADPREQ Req;
269
270 setPathIfConfig();
271
272 switch (argc)
273 {
274 case 5:
275 {
276 /* Add a netmask to existing interface */
277 if (strcmp("netmask", argv[3]))
278 {
279 fprintf(stderr, "Invalid argument: %s\n\n", argv[3]);
280 showUsage();
281 return 1;
282 }
283 pszOption = "netmask";
284 pszNetworkMask = argv[4];
285 pszAdapterName = argv[1];
286 pszAddress = argv[2];
287 break;
288 }
289
290 case 4:
291 {
292 /* Remove a single address from existing interface */
293 if (strcmp("remove", argv[3]))
294 {
295 fprintf(stderr, "Invalid argument: %s\n\n", argv[3]);
296 showUsage();
297 return 1;
298 }
299 fRemove = true;
300 pszAdapterName = argv[1];
301 pszAddress = argv[2];
302 break;
303 }
304
305 case 3:
306 {
307 pszAdapterName = argv[1];
308 memset(&Req, '\0', sizeof(Req));
309#ifdef RT_OS_LINUX
310 if (strcmp("speed", argv[2]) == 0)
311 {
312 /*
313 * This ugly hack is needed for retrieving the link speed on
314 * pre-2.6.33 kernels (see @bugref{6345}).
315 */
316 if (strlen(pszAdapterName) >= IFNAMSIZ)
317 {
318 showUsage();
319 return -1;
320 }
321 struct ifreq IfReq;
322 struct ethtool_value EthToolVal;
323 struct ethtool_cmd EthToolReq;
324 int fd = socket(AF_INET, SOCK_DGRAM, 0);
325 if (fd < 0)
326 {
327 fprintf(stderr, "VBoxNetAdpCtl: Error while retrieving link "
328 "speed for %s: ", pszAdapterName);
329 perror("VBoxNetAdpCtl: failed to open control socket");
330 return ADPCTLERR_SOCKET_FAILED;
331 }
332 /* Get link status first. */
333 memset(&EthToolVal, 0, sizeof(EthToolVal));
334 memset(&IfReq, 0, sizeof(IfReq));
335 snprintf(IfReq.ifr_name, sizeof(IfReq.ifr_name), "%s", pszAdapterName);
336
337 EthToolVal.cmd = ETHTOOL_GLINK;
338 IfReq.ifr_data = (caddr_t)&EthToolVal;
339 rc = ioctl(fd, SIOCETHTOOL, &IfReq);
340 if (rc == 0)
341 {
342 if (EthToolVal.data)
343 {
344 memset(&IfReq, 0, sizeof(IfReq));
345 snprintf(IfReq.ifr_name, sizeof(IfReq.ifr_name), "%s", pszAdapterName);
346 EthToolReq.cmd = ETHTOOL_GSET;
347 IfReq.ifr_data = (caddr_t)&EthToolReq;
348 rc = ioctl(fd, SIOCETHTOOL, &IfReq);
349 if (rc == 0)
350 {
351 printf("%u", EthToolReq.speed);
352 }
353 else
354 {
355 fprintf(stderr, "VBoxNetAdpCtl: Error while retrieving link "
356 "speed for %s: ", pszAdapterName);
357 perror("VBoxNetAdpCtl: ioctl failed");
358 rc = ADPCTLERR_IOCTL_FAILED;
359 }
360 }
361 else
362 printf("0");
363 }
364 else
365 {
366 fprintf(stderr, "VBoxNetAdpCtl: Error while retrieving link "
367 "status for %s: ", pszAdapterName);
368 perror("VBoxNetAdpCtl: ioctl failed");
369 rc = ADPCTLERR_IOCTL_FAILED;
370 }
371
372 close(fd);
373 return rc;
374 }
375#endif
376 rc = checkAdapterName(pszAdapterName, szAdapterName);
377 if (rc)
378 return rc;
379 snprintf(Req.szName, sizeof(Req.szName), "%s", szAdapterName);
380 pszAddress = argv[2];
381 if (strcmp("remove", pszAddress) == 0)
382 {
383 /* Remove an existing interface */
384#ifdef RT_OS_SOLARIS
385 return 1;
386#else
387 return doIOCtl(VBOXNETADP_CTL_REMOVE, &Req);
388#endif
389 }
390 else if (strcmp("add", pszAddress) == 0)
391 {
392 /* Create an interface with given name */
393#ifdef RT_OS_SOLARIS
394 return 1;
395#else
396 rc = doIOCtl(VBOXNETADP_CTL_ADD, &Req);
397 if (rc == 0)
398 puts(Req.szName);
399#endif
400 return rc;
401 }
402 break;
403 }
404
405 case 2:
406 {
407 /* Create a new interface */
408 if (strcmp("add", argv[1]) == 0)
409 {
410#ifdef RT_OS_SOLARIS
411 return 1;
412#else
413 memset(&Req, '\0', sizeof(Req));
414 rc = doIOCtl(VBOXNETADP_CTL_ADD, &Req);
415 if (rc == 0)
416 puts(Req.szName);
417#endif
418 return rc;
419 }
420 }
421 /* fall thru */
422
423 default:
424 fprintf(stderr, "Invalid number of arguments.\n\n");
425 /* fall thru */
426 case 1:
427 showUsage();
428 return 1;
429 }
430
431 rc = checkAdapterName(pszAdapterName, szAdapterName);
432 if (rc)
433 return rc;
434
435 pszAdapterName = szAdapterName;
436
437 if (fRemove)
438 {
439 if (strchr(pszAddress, ':'))
440 rc = executeIfconfig(pszAdapterName, "inet6", VBOXADPCTL_DEL_CMD, pszAddress);
441 else
442 {
443#if defined(RT_OS_LINUX)
444 rc = executeIfconfig(pszAdapterName, "0.0.0.0");
445#else
446 rc = executeIfconfig(pszAdapterName, VBOXADPCTL_DEL_CMD, pszAddress);
447#endif
448
449#ifdef RT_OS_SOLARIS
450 /* On Solaris we can unplumb the ipv4 interface */
451 executeIfconfig(pszAdapterName, "inet", "unplumb");
452#endif
453 }
454 }
455 else
456 {
457 /* We are setting/replacing address. */
458 if (strchr(pszAddress, ':'))
459 {
460#ifdef RT_OS_SOLARIS
461 /* On Solaris we need to plumb the interface first if it's not already plumbed. */
462 if (executeIfconfig(pszAdapterName, "inet6") != 0)
463 executeIfconfig(pszAdapterName, "inet6", "plumb", "up");
464#endif
465 /*
466 * Before we set IPv6 address we'd like to remove
467 * all previously assigned addresses except the
468 * self-assigned one.
469 */
470 if (!removeAddresses(pszAdapterName))
471 rc = EXIT_FAILURE;
472 else
473 rc = executeIfconfig(pszAdapterName, "inet6", VBOXADPCTL_ADD_CMD, pszAddress, pszOption, pszNetworkMask);
474 }
475 else
476 {
477#ifdef RT_OS_SOLARIS
478 /* On Solaris we need to plumb the interface first if it's not already plumbed. */
479 if (executeIfconfig(pszAdapterName, "inet") != 0)
480 executeIfconfig(pszAdapterName, "plumb", "up");
481#endif
482 rc = executeIfconfig(pszAdapterName, pszAddress, pszOption, pszNetworkMask);
483 }
484 }
485 return rc;
486}
487
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