vale-ctl.c revision 260700
1/*
2 * Copyright (C) 2013-2014 Michio Honda. All rights reserved.
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 *   1. Redistributions of source code must retain the above copyright
8 *      notice, this list of conditions and the following disclaimer.
9 *   2. Redistributions in binary form must reproduce the above copyright
10 *      notice, this list of conditions and the following disclaimer in the
11 *    documentation and/or other materials provided with the distribution.
12 *
13 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16 * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23 * SUCH DAMAGE.
24 */
25
26/* $FreeBSD: head/tools/tools/netmap/vale-ctl.c 260700 2014-01-16 00:20:42Z luigi $ */
27
28#include <errno.h>
29#include <stdio.h>
30#include <inttypes.h>	/* PRI* macros */
31#include <string.h>	/* strcmp */
32#include <fcntl.h>	/* open */
33#include <unistd.h>	/* close */
34#include <sys/ioctl.h>	/* ioctl */
35#include <sys/param.h>
36#include <net/if.h>	/* ifreq */
37#include <net/netmap.h>
38#include <net/netmap_user.h>
39#include <libgen.h>	/* basename */
40
41/* debug support */
42#define ND(format, ...)	do {} while(0)
43#define D(format, ...)					\
44	fprintf(stderr, "%s [%d] " format "\n",		\
45	__FUNCTION__, __LINE__, ##__VA_ARGS__)
46
47static int
48bdg_ctl(const char *name, int nr_cmd, int nr_arg)
49{
50	struct nmreq nmr;
51	int error = 0;
52	int fd = open("/dev/netmap", O_RDWR);
53
54	if (fd == -1) {
55		D("Unable to open /dev/netmap");
56		return -1;
57	}
58
59	bzero(&nmr, sizeof(nmr));
60	nmr.nr_version = NETMAP_API;
61	if (name != NULL) /* might be NULL */
62		strncpy(nmr.nr_name, name, sizeof(nmr.nr_name));
63	nmr.nr_cmd = nr_cmd;
64
65	switch (nr_cmd) {
66	case NETMAP_BDG_ATTACH:
67	case NETMAP_BDG_DETACH:
68		if (nr_arg && nr_arg != NETMAP_BDG_HOST)
69			nr_arg = 0;
70		nmr.nr_arg1 = nr_arg;
71		error = ioctl(fd, NIOCREGIF, &nmr);
72		if (error == -1) {
73			ND("Unable to %s %s to the bridge", nr_cmd ==
74			    NETMAP_BDG_DETACH?"detach":"attach", name);
75			perror(name);
76		} else
77			ND("Success to %s %s to the bridge", nr_cmd ==
78			    NETMAP_BDG_DETACH?"detach":"attach", name);
79		break;
80
81	case NETMAP_BDG_LIST:
82		if (strlen(nmr.nr_name)) { /* name to bridge/port info */
83			error = ioctl(fd, NIOCGINFO, &nmr);
84			if (error) {
85				ND("Unable to obtain info for %s", name);
86				perror(name);
87			} else
88				D("%s at bridge:%d port:%d", name, nmr.nr_arg1,
89				    nmr.nr_arg2);
90			break;
91		}
92
93		/* scan all the bridges and ports */
94		nmr.nr_arg1 = nmr.nr_arg2 = 0;
95		for (; !ioctl(fd, NIOCGINFO, &nmr); nmr.nr_arg2++) {
96			D("bridge:%d port:%d %s", nmr.nr_arg1, nmr.nr_arg2,
97			    nmr.nr_name);
98			nmr.nr_name[0] = '\0';
99		}
100
101		break;
102
103	default: /* GINFO */
104		nmr.nr_cmd = nmr.nr_arg1 = nmr.nr_arg2 = 0;
105		error = ioctl(fd, NIOCGINFO, &nmr);
106		if (error) {
107			ND("Unable to get if info for %s", name);
108			perror(name);
109		} else
110			D("%s: %d queues.", name, nmr.nr_rx_rings);
111		break;
112	}
113	close(fd);
114	return error;
115}
116
117int
118main(int argc, char *argv[])
119{
120	int ch, nr_cmd = 0, nr_arg = 0;
121	const char *command = basename(argv[0]);
122	char *name = NULL;
123
124	if (argc > 3) {
125usage:
126		fprintf(stderr,
127			"Usage:\n"
128			"%s arguments\n"
129			"\t-g interface	interface name to get info\n"
130			"\t-d interface	interface name to be detached\n"
131			"\t-a interface	interface name to be attached\n"
132			"\t-h interface	interface name to be attached with the host stack\n"
133			"\t-l list all or specified bridge's interfaces (default)\n"
134			"", command);
135		return 0;
136	}
137
138	while ((ch = getopt(argc, argv, "d:a:h:g:l")) != -1) {
139		name = optarg; /* default */
140		switch (ch) {
141		default:
142			fprintf(stderr, "bad option %c %s", ch, optarg);
143			goto usage;
144		case 'd':
145			nr_cmd = NETMAP_BDG_DETACH;
146			break;
147		case 'a':
148			nr_cmd = NETMAP_BDG_ATTACH;
149			break;
150		case 'h':
151			nr_cmd = NETMAP_BDG_ATTACH;
152			nr_arg = NETMAP_BDG_HOST;
153			break;
154		case 'g':
155			nr_cmd = 0;
156			break;
157		case 'l':
158			nr_cmd = NETMAP_BDG_LIST;
159			if (optind < argc && argv[optind][0] == '-')
160				name = NULL;
161			break;
162		}
163		if (optind != argc) {
164			// fprintf(stderr, "optind %d argc %d\n", optind, argc);
165			goto usage;
166		}
167	}
168	if (argc == 1)
169		nr_cmd = NETMAP_BDG_LIST;
170	return bdg_ctl(name, nr_cmd, nr_arg) ? 1 : 0;
171}
172