vale-ctl.c revision 260700
1251139Sluigi/*
2260368Sluigi * Copyright (C) 2013-2014 Michio Honda. All rights reserved.
3251139Sluigi *
4251139Sluigi * Redistribution and use in source and binary forms, with or without
5251139Sluigi * modification, are permitted provided that the following conditions
6251139Sluigi * are met:
7251139Sluigi *   1. Redistributions of source code must retain the above copyright
8251139Sluigi *      notice, this list of conditions and the following disclaimer.
9251139Sluigi *   2. Redistributions in binary form must reproduce the above copyright
10251139Sluigi *      notice, this list of conditions and the following disclaimer in the
11251139Sluigi *    documentation and/or other materials provided with the distribution.
12251139Sluigi *
13251139Sluigi * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
14251139Sluigi * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15251139Sluigi * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
16251139Sluigi * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
17251139Sluigi * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
18251139Sluigi * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
19251139Sluigi * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
20251139Sluigi * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
21251139Sluigi * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
22251139Sluigi * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
23251139Sluigi * SUCH DAMAGE.
24251139Sluigi */
25251139Sluigi
26251139Sluigi/* $FreeBSD: head/tools/tools/netmap/vale-ctl.c 260700 2014-01-16 00:20:42Z luigi $ */
27251139Sluigi
28251139Sluigi#include <errno.h>
29251139Sluigi#include <stdio.h>
30251139Sluigi#include <inttypes.h>	/* PRI* macros */
31251139Sluigi#include <string.h>	/* strcmp */
32251139Sluigi#include <fcntl.h>	/* open */
33251139Sluigi#include <unistd.h>	/* close */
34251139Sluigi#include <sys/ioctl.h>	/* ioctl */
35251139Sluigi#include <sys/param.h>
36251139Sluigi#include <net/if.h>	/* ifreq */
37251139Sluigi#include <net/netmap.h>
38251139Sluigi#include <net/netmap_user.h>
39251139Sluigi#include <libgen.h>	/* basename */
40251139Sluigi
41251139Sluigi/* debug support */
42251139Sluigi#define ND(format, ...)	do {} while(0)
43251139Sluigi#define D(format, ...)					\
44251139Sluigi	fprintf(stderr, "%s [%d] " format "\n",		\
45251139Sluigi	__FUNCTION__, __LINE__, ##__VA_ARGS__)
46251139Sluigi
47251139Sluigistatic int
48251139Sluigibdg_ctl(const char *name, int nr_cmd, int nr_arg)
49251139Sluigi{
50251139Sluigi	struct nmreq nmr;
51251139Sluigi	int error = 0;
52251139Sluigi	int fd = open("/dev/netmap", O_RDWR);
53251139Sluigi
54251139Sluigi	if (fd == -1) {
55251139Sluigi		D("Unable to open /dev/netmap");
56251139Sluigi		return -1;
57251139Sluigi	}
58251139Sluigi
59251139Sluigi	bzero(&nmr, sizeof(nmr));
60251139Sluigi	nmr.nr_version = NETMAP_API;
61251139Sluigi	if (name != NULL) /* might be NULL */
62251139Sluigi		strncpy(nmr.nr_name, name, sizeof(nmr.nr_name));
63251139Sluigi	nmr.nr_cmd = nr_cmd;
64251139Sluigi
65251139Sluigi	switch (nr_cmd) {
66251139Sluigi	case NETMAP_BDG_ATTACH:
67251139Sluigi	case NETMAP_BDG_DETACH:
68251139Sluigi		if (nr_arg && nr_arg != NETMAP_BDG_HOST)
69251139Sluigi			nr_arg = 0;
70251139Sluigi		nmr.nr_arg1 = nr_arg;
71251139Sluigi		error = ioctl(fd, NIOCREGIF, &nmr);
72260700Sluigi		if (error == -1) {
73260700Sluigi			ND("Unable to %s %s to the bridge", nr_cmd ==
74251139Sluigi			    NETMAP_BDG_DETACH?"detach":"attach", name);
75260700Sluigi			perror(name);
76260700Sluigi		} else
77260700Sluigi			ND("Success to %s %s to the bridge", nr_cmd ==
78251139Sluigi			    NETMAP_BDG_DETACH?"detach":"attach", name);
79251139Sluigi		break;
80251139Sluigi
81251139Sluigi	case NETMAP_BDG_LIST:
82251139Sluigi		if (strlen(nmr.nr_name)) { /* name to bridge/port info */
83251139Sluigi			error = ioctl(fd, NIOCGINFO, &nmr);
84260700Sluigi			if (error) {
85260700Sluigi				ND("Unable to obtain info for %s", name);
86260700Sluigi				perror(name);
87260700Sluigi			} else
88251139Sluigi				D("%s at bridge:%d port:%d", name, nmr.nr_arg1,
89251139Sluigi				    nmr.nr_arg2);
90251139Sluigi			break;
91251139Sluigi		}
92251139Sluigi
93251139Sluigi		/* scan all the bridges and ports */
94251139Sluigi		nmr.nr_arg1 = nmr.nr_arg2 = 0;
95251139Sluigi		for (; !ioctl(fd, NIOCGINFO, &nmr); nmr.nr_arg2++) {
96251139Sluigi			D("bridge:%d port:%d %s", nmr.nr_arg1, nmr.nr_arg2,
97251139Sluigi			    nmr.nr_name);
98251139Sluigi			nmr.nr_name[0] = '\0';
99251139Sluigi		}
100251139Sluigi
101251139Sluigi		break;
102251139Sluigi
103251139Sluigi	default: /* GINFO */
104251139Sluigi		nmr.nr_cmd = nmr.nr_arg1 = nmr.nr_arg2 = 0;
105251139Sluigi		error = ioctl(fd, NIOCGINFO, &nmr);
106260700Sluigi		if (error) {
107260700Sluigi			ND("Unable to get if info for %s", name);
108260700Sluigi			perror(name);
109260700Sluigi		} else
110251139Sluigi			D("%s: %d queues.", name, nmr.nr_rx_rings);
111251139Sluigi		break;
112251139Sluigi	}
113251139Sluigi	close(fd);
114251139Sluigi	return error;
115251139Sluigi}
116251139Sluigi
117251139Sluigiint
118251139Sluigimain(int argc, char *argv[])
119251139Sluigi{
120251139Sluigi	int ch, nr_cmd = 0, nr_arg = 0;
121251139Sluigi	const char *command = basename(argv[0]);
122251139Sluigi	char *name = NULL;
123251139Sluigi
124260368Sluigi	if (argc > 3) {
125251139Sluigiusage:
126251139Sluigi		fprintf(stderr,
127251139Sluigi			"Usage:\n"
128251139Sluigi			"%s arguments\n"
129251139Sluigi			"\t-g interface	interface name to get info\n"
130251139Sluigi			"\t-d interface	interface name to be detached\n"
131251139Sluigi			"\t-a interface	interface name to be attached\n"
132251139Sluigi			"\t-h interface	interface name to be attached with the host stack\n"
133260368Sluigi			"\t-l list all or specified bridge's interfaces (default)\n"
134251139Sluigi			"", command);
135251139Sluigi		return 0;
136251139Sluigi	}
137251139Sluigi
138260368Sluigi	while ((ch = getopt(argc, argv, "d:a:h:g:l")) != -1) {
139260368Sluigi		name = optarg; /* default */
140251139Sluigi		switch (ch) {
141251139Sluigi		default:
142251139Sluigi			fprintf(stderr, "bad option %c %s", ch, optarg);
143251139Sluigi			goto usage;
144251139Sluigi		case 'd':
145251139Sluigi			nr_cmd = NETMAP_BDG_DETACH;
146251139Sluigi			break;
147251139Sluigi		case 'a':
148251139Sluigi			nr_cmd = NETMAP_BDG_ATTACH;
149251139Sluigi			break;
150251139Sluigi		case 'h':
151251139Sluigi			nr_cmd = NETMAP_BDG_ATTACH;
152251139Sluigi			nr_arg = NETMAP_BDG_HOST;
153251139Sluigi			break;
154251139Sluigi		case 'g':
155251139Sluigi			nr_cmd = 0;
156251139Sluigi			break;
157251139Sluigi		case 'l':
158251139Sluigi			nr_cmd = NETMAP_BDG_LIST;
159260368Sluigi			if (optind < argc && argv[optind][0] == '-')
160260368Sluigi				name = NULL;
161251139Sluigi			break;
162251139Sluigi		}
163260368Sluigi		if (optind != argc) {
164260368Sluigi			// fprintf(stderr, "optind %d argc %d\n", optind, argc);
165260368Sluigi			goto usage;
166260368Sluigi		}
167251139Sluigi	}
168251139Sluigi	if (argc == 1)
169251139Sluigi		nr_cmd = NETMAP_BDG_LIST;
170260700Sluigi	return bdg_ctl(name, nr_cmd, nr_arg) ? 1 : 0;
171251139Sluigi}
172