1*de1e4e89SAndroid Build Coastguard Worker /*
2*de1e4e89SAndroid Build Coastguard Worker * iplink_macvlan.c macvlan/macvtap device support
3*de1e4e89SAndroid Build Coastguard Worker *
4*de1e4e89SAndroid Build Coastguard Worker * This program is free software; you can redistribute it and/or
5*de1e4e89SAndroid Build Coastguard Worker * modify it under the terms of the GNU General Public License
6*de1e4e89SAndroid Build Coastguard Worker * as published by the Free Software Foundation; either version
7*de1e4e89SAndroid Build Coastguard Worker * 2 of the License, or (at your option) any later version.
8*de1e4e89SAndroid Build Coastguard Worker *
9*de1e4e89SAndroid Build Coastguard Worker * Authors: Patrick McHardy <[email protected]>
10*de1e4e89SAndroid Build Coastguard Worker * Arnd Bergmann <[email protected]>
11*de1e4e89SAndroid Build Coastguard Worker */
12*de1e4e89SAndroid Build Coastguard Worker
13*de1e4e89SAndroid Build Coastguard Worker #include <stdio.h>
14*de1e4e89SAndroid Build Coastguard Worker #include <stdlib.h>
15*de1e4e89SAndroid Build Coastguard Worker #include <string.h>
16*de1e4e89SAndroid Build Coastguard Worker #include <sys/socket.h>
17*de1e4e89SAndroid Build Coastguard Worker #include <linux/if_link.h>
18*de1e4e89SAndroid Build Coastguard Worker #include <linux/if_ether.h>
19*de1e4e89SAndroid Build Coastguard Worker
20*de1e4e89SAndroid Build Coastguard Worker #include "rt_names.h"
21*de1e4e89SAndroid Build Coastguard Worker #include "utils.h"
22*de1e4e89SAndroid Build Coastguard Worker #include "ip_common.h"
23*de1e4e89SAndroid Build Coastguard Worker
24*de1e4e89SAndroid Build Coastguard Worker #define pfx_err(lu, ...) { \
25*de1e4e89SAndroid Build Coastguard Worker fprintf(stderr, "%s: ", lu->id); \
26*de1e4e89SAndroid Build Coastguard Worker fprintf(stderr, __VA_ARGS__); \
27*de1e4e89SAndroid Build Coastguard Worker fprintf(stderr, "\n"); \
28*de1e4e89SAndroid Build Coastguard Worker }
29*de1e4e89SAndroid Build Coastguard Worker
print_explain(struct link_util * lu,FILE * f)30*de1e4e89SAndroid Build Coastguard Worker static void print_explain(struct link_util *lu, FILE *f)
31*de1e4e89SAndroid Build Coastguard Worker {
32*de1e4e89SAndroid Build Coastguard Worker fprintf(f,
33*de1e4e89SAndroid Build Coastguard Worker "Usage: ... %s mode MODE [flag MODE_FLAG] MODE_OPTS\n"
34*de1e4e89SAndroid Build Coastguard Worker "\n"
35*de1e4e89SAndroid Build Coastguard Worker "MODE: private | vepa | bridge | passthru | source\n"
36*de1e4e89SAndroid Build Coastguard Worker "MODE_FLAG: null | nopromisc\n"
37*de1e4e89SAndroid Build Coastguard Worker "MODE_OPTS: for mode \"source\":\n"
38*de1e4e89SAndroid Build Coastguard Worker "\tmacaddr { { add | del } <macaddr> | set [ <macaddr> [ <macaddr> ... ] ] | flush }\n",
39*de1e4e89SAndroid Build Coastguard Worker lu->id
40*de1e4e89SAndroid Build Coastguard Worker );
41*de1e4e89SAndroid Build Coastguard Worker }
42*de1e4e89SAndroid Build Coastguard Worker
explain(struct link_util * lu)43*de1e4e89SAndroid Build Coastguard Worker static void explain(struct link_util *lu)
44*de1e4e89SAndroid Build Coastguard Worker {
45*de1e4e89SAndroid Build Coastguard Worker print_explain(lu, stderr);
46*de1e4e89SAndroid Build Coastguard Worker }
47*de1e4e89SAndroid Build Coastguard Worker
48*de1e4e89SAndroid Build Coastguard Worker
mode_arg(const char * arg)49*de1e4e89SAndroid Build Coastguard Worker static int mode_arg(const char *arg)
50*de1e4e89SAndroid Build Coastguard Worker {
51*de1e4e89SAndroid Build Coastguard Worker fprintf(stderr,
52*de1e4e89SAndroid Build Coastguard Worker "Error: argument of \"mode\" must be \"private\", \"vepa\", \"bridge\", \"passthru\" or \"source\", not \"%s\"\n",
53*de1e4e89SAndroid Build Coastguard Worker arg);
54*de1e4e89SAndroid Build Coastguard Worker return -1;
55*de1e4e89SAndroid Build Coastguard Worker }
56*de1e4e89SAndroid Build Coastguard Worker
flag_arg(const char * arg)57*de1e4e89SAndroid Build Coastguard Worker static int flag_arg(const char *arg)
58*de1e4e89SAndroid Build Coastguard Worker {
59*de1e4e89SAndroid Build Coastguard Worker fprintf(stderr,
60*de1e4e89SAndroid Build Coastguard Worker "Error: argument of \"flag\" must be \"nopromisc\" or \"null\", not \"%s\"\n",
61*de1e4e89SAndroid Build Coastguard Worker arg);
62*de1e4e89SAndroid Build Coastguard Worker return -1;
63*de1e4e89SAndroid Build Coastguard Worker }
64*de1e4e89SAndroid Build Coastguard Worker
macvlan_parse_opt(struct link_util * lu,int argc,char ** argv,struct nlmsghdr * n)65*de1e4e89SAndroid Build Coastguard Worker static int macvlan_parse_opt(struct link_util *lu, int argc, char **argv,
66*de1e4e89SAndroid Build Coastguard Worker struct nlmsghdr *n)
67*de1e4e89SAndroid Build Coastguard Worker {
68*de1e4e89SAndroid Build Coastguard Worker __u32 mode = 0;
69*de1e4e89SAndroid Build Coastguard Worker __u16 flags = 0;
70*de1e4e89SAndroid Build Coastguard Worker __u32 mac_mode = 0;
71*de1e4e89SAndroid Build Coastguard Worker int has_flags = 0;
72*de1e4e89SAndroid Build Coastguard Worker char mac[ETH_ALEN];
73*de1e4e89SAndroid Build Coastguard Worker struct rtattr *nmac;
74*de1e4e89SAndroid Build Coastguard Worker
75*de1e4e89SAndroid Build Coastguard Worker while (argc > 0) {
76*de1e4e89SAndroid Build Coastguard Worker if (matches(*argv, "mode") == 0) {
77*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG();
78*de1e4e89SAndroid Build Coastguard Worker
79*de1e4e89SAndroid Build Coastguard Worker if (strcmp(*argv, "private") == 0)
80*de1e4e89SAndroid Build Coastguard Worker mode = MACVLAN_MODE_PRIVATE;
81*de1e4e89SAndroid Build Coastguard Worker else if (strcmp(*argv, "vepa") == 0)
82*de1e4e89SAndroid Build Coastguard Worker mode = MACVLAN_MODE_VEPA;
83*de1e4e89SAndroid Build Coastguard Worker else if (strcmp(*argv, "bridge") == 0)
84*de1e4e89SAndroid Build Coastguard Worker mode = MACVLAN_MODE_BRIDGE;
85*de1e4e89SAndroid Build Coastguard Worker else if (strcmp(*argv, "passthru") == 0)
86*de1e4e89SAndroid Build Coastguard Worker mode = MACVLAN_MODE_PASSTHRU;
87*de1e4e89SAndroid Build Coastguard Worker else if (strcmp(*argv, "source") == 0)
88*de1e4e89SAndroid Build Coastguard Worker mode = MACVLAN_MODE_SOURCE;
89*de1e4e89SAndroid Build Coastguard Worker else
90*de1e4e89SAndroid Build Coastguard Worker return mode_arg(*argv);
91*de1e4e89SAndroid Build Coastguard Worker } else if (matches(*argv, "flag") == 0) {
92*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG();
93*de1e4e89SAndroid Build Coastguard Worker
94*de1e4e89SAndroid Build Coastguard Worker if (strcmp(*argv, "nopromisc") == 0)
95*de1e4e89SAndroid Build Coastguard Worker flags |= MACVLAN_FLAG_NOPROMISC;
96*de1e4e89SAndroid Build Coastguard Worker else if (strcmp(*argv, "null") == 0)
97*de1e4e89SAndroid Build Coastguard Worker flags |= 0;
98*de1e4e89SAndroid Build Coastguard Worker else
99*de1e4e89SAndroid Build Coastguard Worker return flag_arg(*argv);
100*de1e4e89SAndroid Build Coastguard Worker
101*de1e4e89SAndroid Build Coastguard Worker has_flags = 1;
102*de1e4e89SAndroid Build Coastguard Worker
103*de1e4e89SAndroid Build Coastguard Worker } else if (matches(*argv, "macaddr") == 0) {
104*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG();
105*de1e4e89SAndroid Build Coastguard Worker
106*de1e4e89SAndroid Build Coastguard Worker if (strcmp(*argv, "add") == 0) {
107*de1e4e89SAndroid Build Coastguard Worker mac_mode = MACVLAN_MACADDR_ADD;
108*de1e4e89SAndroid Build Coastguard Worker } else if (strcmp(*argv, "del") == 0) {
109*de1e4e89SAndroid Build Coastguard Worker mac_mode = MACVLAN_MACADDR_DEL;
110*de1e4e89SAndroid Build Coastguard Worker } else if (strcmp(*argv, "set") == 0) {
111*de1e4e89SAndroid Build Coastguard Worker mac_mode = MACVLAN_MACADDR_SET;
112*de1e4e89SAndroid Build Coastguard Worker } else if (strcmp(*argv, "flush") == 0) {
113*de1e4e89SAndroid Build Coastguard Worker mac_mode = MACVLAN_MACADDR_FLUSH;
114*de1e4e89SAndroid Build Coastguard Worker } else {
115*de1e4e89SAndroid Build Coastguard Worker explain(lu);
116*de1e4e89SAndroid Build Coastguard Worker return -1;
117*de1e4e89SAndroid Build Coastguard Worker }
118*de1e4e89SAndroid Build Coastguard Worker
119*de1e4e89SAndroid Build Coastguard Worker addattr32(n, 1024, IFLA_MACVLAN_MACADDR_MODE, mac_mode);
120*de1e4e89SAndroid Build Coastguard Worker
121*de1e4e89SAndroid Build Coastguard Worker if (mac_mode == MACVLAN_MACADDR_ADD ||
122*de1e4e89SAndroid Build Coastguard Worker mac_mode == MACVLAN_MACADDR_DEL) {
123*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG();
124*de1e4e89SAndroid Build Coastguard Worker
125*de1e4e89SAndroid Build Coastguard Worker if (ll_addr_a2n(mac, sizeof(mac),
126*de1e4e89SAndroid Build Coastguard Worker *argv) != ETH_ALEN)
127*de1e4e89SAndroid Build Coastguard Worker return -1;
128*de1e4e89SAndroid Build Coastguard Worker
129*de1e4e89SAndroid Build Coastguard Worker addattr_l(n, 1024, IFLA_MACVLAN_MACADDR, &mac,
130*de1e4e89SAndroid Build Coastguard Worker ETH_ALEN);
131*de1e4e89SAndroid Build Coastguard Worker }
132*de1e4e89SAndroid Build Coastguard Worker
133*de1e4e89SAndroid Build Coastguard Worker if (mac_mode == MACVLAN_MACADDR_SET) {
134*de1e4e89SAndroid Build Coastguard Worker nmac = addattr_nest(n, 1024,
135*de1e4e89SAndroid Build Coastguard Worker IFLA_MACVLAN_MACADDR_DATA);
136*de1e4e89SAndroid Build Coastguard Worker while (NEXT_ARG_OK()) {
137*de1e4e89SAndroid Build Coastguard Worker NEXT_ARG_FWD();
138*de1e4e89SAndroid Build Coastguard Worker
139*de1e4e89SAndroid Build Coastguard Worker if (ll_addr_a2n(mac, sizeof(mac),
140*de1e4e89SAndroid Build Coastguard Worker *argv) != ETH_ALEN) {
141*de1e4e89SAndroid Build Coastguard Worker PREV_ARG();
142*de1e4e89SAndroid Build Coastguard Worker break;
143*de1e4e89SAndroid Build Coastguard Worker }
144*de1e4e89SAndroid Build Coastguard Worker
145*de1e4e89SAndroid Build Coastguard Worker addattr_l(n, 1024, IFLA_MACVLAN_MACADDR,
146*de1e4e89SAndroid Build Coastguard Worker &mac, ETH_ALEN);
147*de1e4e89SAndroid Build Coastguard Worker }
148*de1e4e89SAndroid Build Coastguard Worker addattr_nest_end(n, nmac);
149*de1e4e89SAndroid Build Coastguard Worker }
150*de1e4e89SAndroid Build Coastguard Worker } else if (matches(*argv, "nopromisc") == 0) {
151*de1e4e89SAndroid Build Coastguard Worker flags |= MACVLAN_FLAG_NOPROMISC;
152*de1e4e89SAndroid Build Coastguard Worker has_flags = 1;
153*de1e4e89SAndroid Build Coastguard Worker } else if (matches(*argv, "help") == 0) {
154*de1e4e89SAndroid Build Coastguard Worker explain(lu);
155*de1e4e89SAndroid Build Coastguard Worker return -1;
156*de1e4e89SAndroid Build Coastguard Worker } else {
157*de1e4e89SAndroid Build Coastguard Worker pfx_err(lu, "unknown option \"%s\"?", *argv);
158*de1e4e89SAndroid Build Coastguard Worker explain(lu);
159*de1e4e89SAndroid Build Coastguard Worker return -1;
160*de1e4e89SAndroid Build Coastguard Worker }
161*de1e4e89SAndroid Build Coastguard Worker argc--, argv++;
162*de1e4e89SAndroid Build Coastguard Worker }
163*de1e4e89SAndroid Build Coastguard Worker
164*de1e4e89SAndroid Build Coastguard Worker if (mode)
165*de1e4e89SAndroid Build Coastguard Worker addattr32(n, 1024, IFLA_MACVLAN_MODE, mode);
166*de1e4e89SAndroid Build Coastguard Worker
167*de1e4e89SAndroid Build Coastguard Worker if (has_flags) {
168*de1e4e89SAndroid Build Coastguard Worker if (flags & MACVLAN_FLAG_NOPROMISC &&
169*de1e4e89SAndroid Build Coastguard Worker mode != MACVLAN_MODE_PASSTHRU) {
170*de1e4e89SAndroid Build Coastguard Worker pfx_err(lu, "nopromisc flag only valid in passthru mode");
171*de1e4e89SAndroid Build Coastguard Worker explain(lu);
172*de1e4e89SAndroid Build Coastguard Worker return -1;
173*de1e4e89SAndroid Build Coastguard Worker }
174*de1e4e89SAndroid Build Coastguard Worker addattr16(n, 1024, IFLA_MACVLAN_FLAGS, flags);
175*de1e4e89SAndroid Build Coastguard Worker }
176*de1e4e89SAndroid Build Coastguard Worker return 0;
177*de1e4e89SAndroid Build Coastguard Worker }
178*de1e4e89SAndroid Build Coastguard Worker
macvlan_print_opt(struct link_util * lu,FILE * f,struct rtattr * tb[])179*de1e4e89SAndroid Build Coastguard Worker static void macvlan_print_opt(struct link_util *lu, FILE *f, struct rtattr *tb[])
180*de1e4e89SAndroid Build Coastguard Worker {
181*de1e4e89SAndroid Build Coastguard Worker __u32 mode;
182*de1e4e89SAndroid Build Coastguard Worker __u16 flags;
183*de1e4e89SAndroid Build Coastguard Worker __u32 count;
184*de1e4e89SAndroid Build Coastguard Worker unsigned char *addr;
185*de1e4e89SAndroid Build Coastguard Worker int len;
186*de1e4e89SAndroid Build Coastguard Worker struct rtattr *rta;
187*de1e4e89SAndroid Build Coastguard Worker
188*de1e4e89SAndroid Build Coastguard Worker if (!tb)
189*de1e4e89SAndroid Build Coastguard Worker return;
190*de1e4e89SAndroid Build Coastguard Worker
191*de1e4e89SAndroid Build Coastguard Worker if (!tb[IFLA_MACVLAN_MODE] ||
192*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(tb[IFLA_MACVLAN_MODE]) < sizeof(__u32))
193*de1e4e89SAndroid Build Coastguard Worker return;
194*de1e4e89SAndroid Build Coastguard Worker
195*de1e4e89SAndroid Build Coastguard Worker mode = rta_getattr_u32(tb[IFLA_MACVLAN_MODE]);
196*de1e4e89SAndroid Build Coastguard Worker print_string(PRINT_ANY,
197*de1e4e89SAndroid Build Coastguard Worker "mode",
198*de1e4e89SAndroid Build Coastguard Worker "mode %s ",
199*de1e4e89SAndroid Build Coastguard Worker mode == MACVLAN_MODE_PRIVATE ? "private"
200*de1e4e89SAndroid Build Coastguard Worker : mode == MACVLAN_MODE_VEPA ? "vepa"
201*de1e4e89SAndroid Build Coastguard Worker : mode == MACVLAN_MODE_BRIDGE ? "bridge"
202*de1e4e89SAndroid Build Coastguard Worker : mode == MACVLAN_MODE_PASSTHRU ? "passthru"
203*de1e4e89SAndroid Build Coastguard Worker : mode == MACVLAN_MODE_SOURCE ? "source"
204*de1e4e89SAndroid Build Coastguard Worker : "unknown");
205*de1e4e89SAndroid Build Coastguard Worker
206*de1e4e89SAndroid Build Coastguard Worker if (!tb[IFLA_MACVLAN_FLAGS] ||
207*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(tb[IFLA_MACVLAN_FLAGS]) < sizeof(__u16))
208*de1e4e89SAndroid Build Coastguard Worker flags = 0;
209*de1e4e89SAndroid Build Coastguard Worker else
210*de1e4e89SAndroid Build Coastguard Worker flags = rta_getattr_u16(tb[IFLA_MACVLAN_FLAGS]);
211*de1e4e89SAndroid Build Coastguard Worker
212*de1e4e89SAndroid Build Coastguard Worker if (flags & MACVLAN_FLAG_NOPROMISC)
213*de1e4e89SAndroid Build Coastguard Worker print_bool(PRINT_ANY, "nopromisc", "nopromisc ", true);
214*de1e4e89SAndroid Build Coastguard Worker
215*de1e4e89SAndroid Build Coastguard Worker /* in source mode, there are more options to print */
216*de1e4e89SAndroid Build Coastguard Worker
217*de1e4e89SAndroid Build Coastguard Worker if (mode != MACVLAN_MODE_SOURCE)
218*de1e4e89SAndroid Build Coastguard Worker return;
219*de1e4e89SAndroid Build Coastguard Worker
220*de1e4e89SAndroid Build Coastguard Worker if (!tb[IFLA_MACVLAN_MACADDR_COUNT] ||
221*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(tb[IFLA_MACVLAN_MACADDR_COUNT]) < sizeof(__u32))
222*de1e4e89SAndroid Build Coastguard Worker return;
223*de1e4e89SAndroid Build Coastguard Worker
224*de1e4e89SAndroid Build Coastguard Worker count = rta_getattr_u32(tb[IFLA_MACVLAN_MACADDR_COUNT]);
225*de1e4e89SAndroid Build Coastguard Worker print_int(PRINT_ANY, "macaddr_count", "remotes (%d) ", count);
226*de1e4e89SAndroid Build Coastguard Worker
227*de1e4e89SAndroid Build Coastguard Worker if (!tb[IFLA_MACVLAN_MACADDR_DATA])
228*de1e4e89SAndroid Build Coastguard Worker return;
229*de1e4e89SAndroid Build Coastguard Worker
230*de1e4e89SAndroid Build Coastguard Worker rta = RTA_DATA(tb[IFLA_MACVLAN_MACADDR_DATA]);
231*de1e4e89SAndroid Build Coastguard Worker len = RTA_PAYLOAD(tb[IFLA_MACVLAN_MACADDR_DATA]);
232*de1e4e89SAndroid Build Coastguard Worker
233*de1e4e89SAndroid Build Coastguard Worker open_json_array(PRINT_JSON, "macaddr_data");
234*de1e4e89SAndroid Build Coastguard Worker for (; RTA_OK(rta, len); rta = RTA_NEXT(rta, len)) {
235*de1e4e89SAndroid Build Coastguard Worker if (rta->rta_type != IFLA_MACVLAN_MACADDR ||
236*de1e4e89SAndroid Build Coastguard Worker RTA_PAYLOAD(rta) < 6)
237*de1e4e89SAndroid Build Coastguard Worker continue;
238*de1e4e89SAndroid Build Coastguard Worker addr = RTA_DATA(rta);
239*de1e4e89SAndroid Build Coastguard Worker if (is_json_context()) {
240*de1e4e89SAndroid Build Coastguard Worker SPRINT_BUF(b1);
241*de1e4e89SAndroid Build Coastguard Worker
242*de1e4e89SAndroid Build Coastguard Worker snprintf(b1, sizeof(b1),
243*de1e4e89SAndroid Build Coastguard Worker "%.2x:%.2x:%.2x:%.2x:%.2x:%.2x", addr[0],
244*de1e4e89SAndroid Build Coastguard Worker addr[1], addr[2], addr[3], addr[4], addr[5]);
245*de1e4e89SAndroid Build Coastguard Worker print_string(PRINT_JSON, NULL, NULL, b1);
246*de1e4e89SAndroid Build Coastguard Worker } else {
247*de1e4e89SAndroid Build Coastguard Worker fprintf(f, "%.2x:%.2x:%.2x:%.2x:%.2x:%.2x ", addr[0],
248*de1e4e89SAndroid Build Coastguard Worker addr[1], addr[2], addr[3], addr[4], addr[5]);
249*de1e4e89SAndroid Build Coastguard Worker }
250*de1e4e89SAndroid Build Coastguard Worker }
251*de1e4e89SAndroid Build Coastguard Worker close_json_array(PRINT_JSON, NULL);
252*de1e4e89SAndroid Build Coastguard Worker }
253*de1e4e89SAndroid Build Coastguard Worker
macvlan_print_help(struct link_util * lu,int argc,char ** argv,FILE * f)254*de1e4e89SAndroid Build Coastguard Worker static void macvlan_print_help(struct link_util *lu, int argc, char **argv,
255*de1e4e89SAndroid Build Coastguard Worker FILE *f)
256*de1e4e89SAndroid Build Coastguard Worker {
257*de1e4e89SAndroid Build Coastguard Worker print_explain(lu, f);
258*de1e4e89SAndroid Build Coastguard Worker }
259*de1e4e89SAndroid Build Coastguard Worker
260*de1e4e89SAndroid Build Coastguard Worker struct link_util macvlan_link_util = {
261*de1e4e89SAndroid Build Coastguard Worker .id = "macvlan",
262*de1e4e89SAndroid Build Coastguard Worker .maxattr = IFLA_MACVLAN_MAX,
263*de1e4e89SAndroid Build Coastguard Worker .parse_opt = macvlan_parse_opt,
264*de1e4e89SAndroid Build Coastguard Worker .print_opt = macvlan_print_opt,
265*de1e4e89SAndroid Build Coastguard Worker .print_help = macvlan_print_help,
266*de1e4e89SAndroid Build Coastguard Worker };
267*de1e4e89SAndroid Build Coastguard Worker
268*de1e4e89SAndroid Build Coastguard Worker struct link_util macvtap_link_util = {
269*de1e4e89SAndroid Build Coastguard Worker .id = "macvtap",
270*de1e4e89SAndroid Build Coastguard Worker .maxattr = IFLA_MACVLAN_MAX,
271*de1e4e89SAndroid Build Coastguard Worker .parse_opt = macvlan_parse_opt,
272*de1e4e89SAndroid Build Coastguard Worker .print_opt = macvlan_print_opt,
273*de1e4e89SAndroid Build Coastguard Worker .print_help = macvlan_print_help,
274*de1e4e89SAndroid Build Coastguard Worker };
275