1*cf5a6c84SAndroid Build Coastguard Worker /* ip.c - Show / manipulate routing, devices, policy routing and tunnels.
2*cf5a6c84SAndroid Build Coastguard Worker *
3*cf5a6c84SAndroid Build Coastguard Worker * Copyright 2014 Sameer Prakash Pradhan <[email protected]>
4*cf5a6c84SAndroid Build Coastguard Worker * Copyright 2014 Ranjan Kumar <[email protected]>
5*cf5a6c84SAndroid Build Coastguard Worker * Copyright 2014 Rajni Kant <[email protected]>
6*cf5a6c84SAndroid Build Coastguard Worker * Copyright 2014 Bilal Qureshi <[email protected]>
7*cf5a6c84SAndroid Build Coastguard Worker *
8*cf5a6c84SAndroid Build Coastguard Worker * No Standard.
9*cf5a6c84SAndroid Build Coastguard Worker *
10*cf5a6c84SAndroid Build Coastguard Worker USE_IP(NEWTOY(ip, NULL, TOYFLAG_SBIN))
11*cf5a6c84SAndroid Build Coastguard Worker USE_IP(OLDTOY(ipaddr, ip, TOYFLAG_SBIN))
12*cf5a6c84SAndroid Build Coastguard Worker USE_IP(OLDTOY(iplink, ip, TOYFLAG_SBIN))
13*cf5a6c84SAndroid Build Coastguard Worker USE_IP(OLDTOY(iproute, ip, TOYFLAG_SBIN))
14*cf5a6c84SAndroid Build Coastguard Worker USE_IP(OLDTOY(iprule, ip, TOYFLAG_SBIN))
15*cf5a6c84SAndroid Build Coastguard Worker USE_IP(OLDTOY(iptunnel, ip, TOYFLAG_SBIN))
16*cf5a6c84SAndroid Build Coastguard Worker
17*cf5a6c84SAndroid Build Coastguard Worker config IP
18*cf5a6c84SAndroid Build Coastguard Worker bool "ip"
19*cf5a6c84SAndroid Build Coastguard Worker default n
20*cf5a6c84SAndroid Build Coastguard Worker help
21*cf5a6c84SAndroid Build Coastguard Worker usage: ip [ OPTIONS ] OBJECT { COMMAND }
22*cf5a6c84SAndroid Build Coastguard Worker
23*cf5a6c84SAndroid Build Coastguard Worker Show / manipulate routing, devices, policy routing and tunnels.
24*cf5a6c84SAndroid Build Coastguard Worker
25*cf5a6c84SAndroid Build Coastguard Worker where OBJECT := {address | link | route | rule | tunnel}
26*cf5a6c84SAndroid Build Coastguard Worker OPTIONS := { -f[amily] { inet | inet6 | link } | -o[neline] }
27*cf5a6c84SAndroid Build Coastguard Worker */
28*cf5a6c84SAndroid Build Coastguard Worker #define FOR_ip
29*cf5a6c84SAndroid Build Coastguard Worker #include "toys.h"
30*cf5a6c84SAndroid Build Coastguard Worker #include <linux/netlink.h>
31*cf5a6c84SAndroid Build Coastguard Worker #include <linux/rtnetlink.h>
32*cf5a6c84SAndroid Build Coastguard Worker #include <linux/if_ether.h>
33*cf5a6c84SAndroid Build Coastguard Worker #include <linux/if_addr.h>
34*cf5a6c84SAndroid Build Coastguard Worker #include <net/if_arp.h>
35*cf5a6c84SAndroid Build Coastguard Worker #include <ifaddrs.h>
36*cf5a6c84SAndroid Build Coastguard Worker #include <fnmatch.h>
37*cf5a6c84SAndroid Build Coastguard Worker #include <linux/ip.h> // Centos 7.2 EOL June 30 2024
38*cf5a6c84SAndroid Build Coastguard Worker #include <linux/if_tunnel.h>
39*cf5a6c84SAndroid Build Coastguard Worker
40*cf5a6c84SAndroid Build Coastguard Worker #ifndef IP_DF
41*cf5a6c84SAndroid Build Coastguard Worker #define IP_DF 0x4000 /* don't fragment flag. */
42*cf5a6c84SAndroid Build Coastguard Worker #endif
43*cf5a6c84SAndroid Build Coastguard Worker
44*cf5a6c84SAndroid Build Coastguard Worker GLOBALS(
45*cf5a6c84SAndroid Build Coastguard Worker char stats, singleline, flush, *filter_dev, gbuf[8192];
46*cf5a6c84SAndroid Build Coastguard Worker int sockfd, connected, from_ok, route_cmd;
47*cf5a6c84SAndroid Build Coastguard Worker int8_t addressfamily, is_addr;
48*cf5a6c84SAndroid Build Coastguard Worker )
49*cf5a6c84SAndroid Build Coastguard Worker
50*cf5a6c84SAndroid Build Coastguard Worker struct arglist {
51*cf5a6c84SAndroid Build Coastguard Worker char *name;
52*cf5a6c84SAndroid Build Coastguard Worker int idx;
53*cf5a6c84SAndroid Build Coastguard Worker };
54*cf5a6c84SAndroid Build Coastguard Worker
55*cf5a6c84SAndroid Build Coastguard Worker static struct
56*cf5a6c84SAndroid Build Coastguard Worker {
57*cf5a6c84SAndroid Build Coastguard Worker int ifindex, scope, scopemask, up, to;
58*cf5a6c84SAndroid Build Coastguard Worker char *label, *addr;
59*cf5a6c84SAndroid Build Coastguard Worker } addrinfo;
60*cf5a6c84SAndroid Build Coastguard Worker
61*cf5a6c84SAndroid Build Coastguard Worker struct linkdata {
62*cf5a6c84SAndroid Build Coastguard Worker struct linkdata *next, *prev;
63*cf5a6c84SAndroid Build Coastguard Worker int flags, iface_idx, mtu, txqueuelen, parent,iface_type;
64*cf5a6c84SAndroid Build Coastguard Worker char qdiscpline[IFNAMSIZ+1], state[IFNAMSIZ+1], type[IFNAMSIZ+1],
65*cf5a6c84SAndroid Build Coastguard Worker iface[IFNAMSIZ+1], laddr[64], bcast[64];
66*cf5a6c84SAndroid Build Coastguard Worker struct rtnl_link_stats rt_stat;
67*cf5a6c84SAndroid Build Coastguard Worker }*linfo;
68*cf5a6c84SAndroid Build Coastguard Worker
69*cf5a6c84SAndroid Build Coastguard Worker typedef int (*cmdobj)(char **argv);
70*cf5a6c84SAndroid Build Coastguard Worker
71*cf5a6c84SAndroid Build Coastguard Worker #define MESG_LEN 8192
72*cf5a6c84SAndroid Build Coastguard Worker
73*cf5a6c84SAndroid Build Coastguard Worker // For "/etc/iproute2/RPDB_tables"
74*cf5a6c84SAndroid Build Coastguard Worker enum {
75*cf5a6c84SAndroid Build Coastguard Worker RPDB_rtdsfield = 1,
76*cf5a6c84SAndroid Build Coastguard Worker RPDB_rtprotos = 2,
77*cf5a6c84SAndroid Build Coastguard Worker RPDB_rtrealms = 3,
78*cf5a6c84SAndroid Build Coastguard Worker RPDB_rtscopes = 4,
79*cf5a6c84SAndroid Build Coastguard Worker RPDB_rttables = 5
80*cf5a6c84SAndroid Build Coastguard Worker };
81*cf5a6c84SAndroid Build Coastguard Worker
82*cf5a6c84SAndroid Build Coastguard Worker #define RPDB_ENTRIES 256
83*cf5a6c84SAndroid Build Coastguard Worker static int8_t rttable_init;
84*cf5a6c84SAndroid Build Coastguard Worker static int8_t rtprotos_init;
85*cf5a6c84SAndroid Build Coastguard Worker static int8_t rtdsfield_init;
86*cf5a6c84SAndroid Build Coastguard Worker static int8_t rtscope_init;
87*cf5a6c84SAndroid Build Coastguard Worker static int8_t rtrealms_init;
88*cf5a6c84SAndroid Build Coastguard Worker
89*cf5a6c84SAndroid Build Coastguard Worker static struct arglist *rt_dsfield[RPDB_ENTRIES];
90*cf5a6c84SAndroid Build Coastguard Worker static struct arglist *rt_protos[RPDB_ENTRIES];
91*cf5a6c84SAndroid Build Coastguard Worker static struct arglist *rt_tables[RPDB_ENTRIES];
92*cf5a6c84SAndroid Build Coastguard Worker static struct arglist *rt_realms[RPDB_ENTRIES];
93*cf5a6c84SAndroid Build Coastguard Worker static struct arglist *rt_scope[RPDB_ENTRIES];
94*cf5a6c84SAndroid Build Coastguard Worker
95*cf5a6c84SAndroid Build Coastguard Worker static struct arglist rtmtypes[] = { {"none", RTN_UNSPEC},
96*cf5a6c84SAndroid Build Coastguard Worker {"unicast", RTN_UNICAST}, {"local", RTN_LOCAL},
97*cf5a6c84SAndroid Build Coastguard Worker {"broadcast", RTN_BROADCAST}, {"anycast", RTN_ANYCAST},
98*cf5a6c84SAndroid Build Coastguard Worker {"multicast", RTN_MULTICAST}, {"blackhole", RTN_BLACKHOLE},
99*cf5a6c84SAndroid Build Coastguard Worker {"unreachable", RTN_UNREACHABLE}, {"prohibit", RTN_PROHIBIT},
100*cf5a6c84SAndroid Build Coastguard Worker {"throw", RTN_THROW}, {"nat", RTN_NAT},
101*cf5a6c84SAndroid Build Coastguard Worker {"xresolve", RTN_XRESOLVE}, {NULL, -1}
102*cf5a6c84SAndroid Build Coastguard Worker };
103*cf5a6c84SAndroid Build Coastguard Worker
104*cf5a6c84SAndroid Build Coastguard Worker static int filter_nlmesg(int (*fun)(struct nlmsghdr *mhdr, char **), char **);
105*cf5a6c84SAndroid Build Coastguard Worker static int ipaddr_print(struct linkdata *, int flg);
106*cf5a6c84SAndroid Build Coastguard Worker
107*cf5a6c84SAndroid Build Coastguard Worker // extended route attribute metrics.
108*cf5a6c84SAndroid Build Coastguard Worker static const char *mx_names[RTAX_MAX + 1] = {
109*cf5a6c84SAndroid Build Coastguard Worker [RTAX_MTU] = "mtu",
110*cf5a6c84SAndroid Build Coastguard Worker [RTAX_WINDOW] = "window",
111*cf5a6c84SAndroid Build Coastguard Worker [RTAX_RTT] = "rtt",
112*cf5a6c84SAndroid Build Coastguard Worker [RTAX_RTTVAR] = "rttvar",
113*cf5a6c84SAndroid Build Coastguard Worker [RTAX_SSTHRESH] = "ssthresh",
114*cf5a6c84SAndroid Build Coastguard Worker [RTAX_CWND] = "cwnd",
115*cf5a6c84SAndroid Build Coastguard Worker [RTAX_ADVMSS] = "advmss",
116*cf5a6c84SAndroid Build Coastguard Worker [RTAX_REORDERING] = "reordering",
117*cf5a6c84SAndroid Build Coastguard Worker [RTAX_HOPLIMIT] = "hoplimit",
118*cf5a6c84SAndroid Build Coastguard Worker [RTAX_INITCWND] = "initcwnd",
119*cf5a6c84SAndroid Build Coastguard Worker [RTAX_FEATURES] = "features",
120*cf5a6c84SAndroid Build Coastguard Worker [RTAX_RTO_MIN] = "rto_min",
121*cf5a6c84SAndroid Build Coastguard Worker [RTAX_INITRWND] = "initrwnd",
122*cf5a6c84SAndroid Build Coastguard Worker [RTAX_QUICKACK] = "quickack",
123*cf5a6c84SAndroid Build Coastguard Worker [RTAX_CC_ALGO] = "congctl"};
124*cf5a6c84SAndroid Build Coastguard Worker
125*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
126*cf5a6c84SAndroid Build Coastguard Worker // Common Code for IP Options (like: addr, link, route etc.)
127*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
substring_to_idx(char * str,struct arglist * list)128*cf5a6c84SAndroid Build Coastguard Worker static int substring_to_idx(char *str, struct arglist *list)
129*cf5a6c84SAndroid Build Coastguard Worker {
130*cf5a6c84SAndroid Build Coastguard Worker struct arglist *alist;
131*cf5a6c84SAndroid Build Coastguard Worker int len;
132*cf5a6c84SAndroid Build Coastguard Worker
133*cf5a6c84SAndroid Build Coastguard Worker if (!str) return -1;
134*cf5a6c84SAndroid Build Coastguard Worker len = strlen(str);
135*cf5a6c84SAndroid Build Coastguard Worker
136*cf5a6c84SAndroid Build Coastguard Worker for (alist = list; alist->name; alist++)
137*cf5a6c84SAndroid Build Coastguard Worker if (!memcmp(str, alist->name, len)) return alist->idx;
138*cf5a6c84SAndroid Build Coastguard Worker return -1;
139*cf5a6c84SAndroid Build Coastguard Worker }
140*cf5a6c84SAndroid Build Coastguard Worker
string_to_idx(char * str,struct arglist * list)141*cf5a6c84SAndroid Build Coastguard Worker static int string_to_idx(char *str, struct arglist *list)
142*cf5a6c84SAndroid Build Coastguard Worker {
143*cf5a6c84SAndroid Build Coastguard Worker struct arglist *alist;
144*cf5a6c84SAndroid Build Coastguard Worker
145*cf5a6c84SAndroid Build Coastguard Worker if (!str) return -1;
146*cf5a6c84SAndroid Build Coastguard Worker for (alist = list; alist->name; alist++)
147*cf5a6c84SAndroid Build Coastguard Worker if (!strcmp(str, alist->name)) return alist->idx;
148*cf5a6c84SAndroid Build Coastguard Worker return -1;
149*cf5a6c84SAndroid Build Coastguard Worker }
150*cf5a6c84SAndroid Build Coastguard Worker
idx_to_string(int idx,struct arglist * list)151*cf5a6c84SAndroid Build Coastguard Worker static char *idx_to_string(int idx, struct arglist *list)
152*cf5a6c84SAndroid Build Coastguard Worker {
153*cf5a6c84SAndroid Build Coastguard Worker struct arglist *alist;
154*cf5a6c84SAndroid Build Coastguard Worker
155*cf5a6c84SAndroid Build Coastguard Worker if (idx < 0) return NULL;
156*cf5a6c84SAndroid Build Coastguard Worker for (alist = list; alist->name; alist++)
157*cf5a6c84SAndroid Build Coastguard Worker if (idx == alist->idx) return alist->name;
158*cf5a6c84SAndroid Build Coastguard Worker return NULL;
159*cf5a6c84SAndroid Build Coastguard Worker }
160*cf5a6c84SAndroid Build Coastguard Worker
send_nlmesg(int type,int flags,int family,void * buf,int blen)161*cf5a6c84SAndroid Build Coastguard Worker static void send_nlmesg(int type, int flags, int family,
162*cf5a6c84SAndroid Build Coastguard Worker void *buf, int blen)
163*cf5a6c84SAndroid Build Coastguard Worker {
164*cf5a6c84SAndroid Build Coastguard Worker struct {
165*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr nlh;
166*cf5a6c84SAndroid Build Coastguard Worker struct rtgenmsg g;
167*cf5a6c84SAndroid Build Coastguard Worker } req;
168*cf5a6c84SAndroid Build Coastguard Worker
169*cf5a6c84SAndroid Build Coastguard Worker if (!buf) {
170*cf5a6c84SAndroid Build Coastguard Worker memset(&req, 0, sizeof(req));
171*cf5a6c84SAndroid Build Coastguard Worker req.nlh.nlmsg_len = sizeof(req);
172*cf5a6c84SAndroid Build Coastguard Worker req.nlh.nlmsg_type = type;
173*cf5a6c84SAndroid Build Coastguard Worker req.nlh.nlmsg_flags = flags;
174*cf5a6c84SAndroid Build Coastguard Worker req.g.rtgen_family = family;
175*cf5a6c84SAndroid Build Coastguard Worker buf = &req;
176*cf5a6c84SAndroid Build Coastguard Worker blen = sizeof(req);
177*cf5a6c84SAndroid Build Coastguard Worker }
178*cf5a6c84SAndroid Build Coastguard Worker if (send(TT.sockfd , (void*)buf, blen, 0) < 0)
179*cf5a6c84SAndroid Build Coastguard Worker perror_exit("Unable to send data on socket.");
180*cf5a6c84SAndroid Build Coastguard Worker }
181*cf5a6c84SAndroid Build Coastguard Worker
182*cf5a6c84SAndroid Build Coastguard Worker // Parse /etc/iproute2/RPDB_tables and prepare list.
parseRPDB(char * fname,struct arglist ** list,int32_t size)183*cf5a6c84SAndroid Build Coastguard Worker static void parseRPDB(char *fname, struct arglist **list, int32_t size)
184*cf5a6c84SAndroid Build Coastguard Worker {
185*cf5a6c84SAndroid Build Coastguard Worker FILE *fp = fopen(fname, "r");
186*cf5a6c84SAndroid Build Coastguard Worker char *line = 0;
187*cf5a6c84SAndroid Build Coastguard Worker size_t l = 0;
188*cf5a6c84SAndroid Build Coastguard Worker ssize_t len;
189*cf5a6c84SAndroid Build Coastguard Worker
190*cf5a6c84SAndroid Build Coastguard Worker if (!fp) return;
191*cf5a6c84SAndroid Build Coastguard Worker while ((len = getline(&line, &l, fp)) > 0) {
192*cf5a6c84SAndroid Build Coastguard Worker char *ptr = line;
193*cf5a6c84SAndroid Build Coastguard Worker int32_t idx;
194*cf5a6c84SAndroid Build Coastguard Worker
195*cf5a6c84SAndroid Build Coastguard Worker while (*ptr == ' ' || *ptr == '\t') ptr++;
196*cf5a6c84SAndroid Build Coastguard Worker if (*ptr == 0 || *ptr == '#' || *ptr == '\n') continue;
197*cf5a6c84SAndroid Build Coastguard Worker if ((sscanf(ptr, "0x%x %s\n", &idx, toybuf) != 2) &&
198*cf5a6c84SAndroid Build Coastguard Worker (sscanf(ptr, "0x%x %s #", &idx, toybuf) != 2) &&
199*cf5a6c84SAndroid Build Coastguard Worker (sscanf(ptr, "%d %s\n", &idx, toybuf) != 2) &&
200*cf5a6c84SAndroid Build Coastguard Worker (sscanf(ptr, "%d %s #", &idx, toybuf) != 2)) {
201*cf5a6c84SAndroid Build Coastguard Worker error_msg("corrupt %s", fname);
202*cf5a6c84SAndroid Build Coastguard Worker break;
203*cf5a6c84SAndroid Build Coastguard Worker }
204*cf5a6c84SAndroid Build Coastguard Worker if (idx >= 0 && idx < size) {
205*cf5a6c84SAndroid Build Coastguard Worker int index = idx & (size-1);
206*cf5a6c84SAndroid Build Coastguard Worker if (list[index]) free(list[index]->name);
207*cf5a6c84SAndroid Build Coastguard Worker else list[index] = xzalloc(sizeof(struct arglist));
208*cf5a6c84SAndroid Build Coastguard Worker list[index]->idx = idx;
209*cf5a6c84SAndroid Build Coastguard Worker list[index]->name = xstrdup(toybuf);
210*cf5a6c84SAndroid Build Coastguard Worker }
211*cf5a6c84SAndroid Build Coastguard Worker }
212*cf5a6c84SAndroid Build Coastguard Worker free(line);
213*cf5a6c84SAndroid Build Coastguard Worker fclose(fp);
214*cf5a6c84SAndroid Build Coastguard Worker }
215*cf5a6c84SAndroid Build Coastguard Worker
free_alist(struct arglist ** list)216*cf5a6c84SAndroid Build Coastguard Worker static void free_alist(struct arglist **list)
217*cf5a6c84SAndroid Build Coastguard Worker {
218*cf5a6c84SAndroid Build Coastguard Worker int i;
219*cf5a6c84SAndroid Build Coastguard Worker for (i = 0;i<RPDB_ENTRIES;i++) {
220*cf5a6c84SAndroid Build Coastguard Worker if (list[i]) {
221*cf5a6c84SAndroid Build Coastguard Worker free(list[i]->name);
222*cf5a6c84SAndroid Build Coastguard Worker free(list[i]);
223*cf5a6c84SAndroid Build Coastguard Worker }
224*cf5a6c84SAndroid Build Coastguard Worker }
225*cf5a6c84SAndroid Build Coastguard Worker }
226*cf5a6c84SAndroid Build Coastguard Worker
init_arglist(struct arglist ** list,int value,char * name)227*cf5a6c84SAndroid Build Coastguard Worker static void init_arglist(struct arglist **list,int value, char* name)
228*cf5a6c84SAndroid Build Coastguard Worker {
229*cf5a6c84SAndroid Build Coastguard Worker if (!list[value]) list[value] = xzalloc(sizeof(struct arglist));
230*cf5a6c84SAndroid Build Coastguard Worker list[value]->idx = value;
231*cf5a6c84SAndroid Build Coastguard Worker list[value]->name = xstrdup(name);
232*cf5a6c84SAndroid Build Coastguard Worker }
233*cf5a6c84SAndroid Build Coastguard Worker
getlist(u_int8_t whichDB)234*cf5a6c84SAndroid Build Coastguard Worker static struct arglist **getlist(u_int8_t whichDB)
235*cf5a6c84SAndroid Build Coastguard Worker {
236*cf5a6c84SAndroid Build Coastguard Worker struct arglist **alist;
237*cf5a6c84SAndroid Build Coastguard Worker
238*cf5a6c84SAndroid Build Coastguard Worker switch (whichDB) {
239*cf5a6c84SAndroid Build Coastguard Worker case RPDB_rtdsfield:
240*cf5a6c84SAndroid Build Coastguard Worker alist = rt_dsfield;
241*cf5a6c84SAndroid Build Coastguard Worker if (!rtdsfield_init) {
242*cf5a6c84SAndroid Build Coastguard Worker rtdsfield_init = 1;
243*cf5a6c84SAndroid Build Coastguard Worker parseRPDB("/etc/iproute2/rt_dsfield", alist, ARRAY_LEN(rt_dsfield));
244*cf5a6c84SAndroid Build Coastguard Worker }
245*cf5a6c84SAndroid Build Coastguard Worker break;
246*cf5a6c84SAndroid Build Coastguard Worker case RPDB_rtprotos:
247*cf5a6c84SAndroid Build Coastguard Worker alist = rt_protos;
248*cf5a6c84SAndroid Build Coastguard Worker if (!rttable_init) {
249*cf5a6c84SAndroid Build Coastguard Worker rtprotos_init = 1;
250*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,0,"none");
251*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,1,"redirect");
252*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,2,"kernel");
253*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,3,"boot");
254*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,4,"static");
255*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,8,"gated");
256*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,9,"ra");
257*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,10,"mrt");
258*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,11,"zebra");
259*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_protos,12,"bird");
260*cf5a6c84SAndroid Build Coastguard Worker parseRPDB("/etc/iproute2/rt_protos", alist, ARRAY_LEN(rt_protos));
261*cf5a6c84SAndroid Build Coastguard Worker }
262*cf5a6c84SAndroid Build Coastguard Worker break;
263*cf5a6c84SAndroid Build Coastguard Worker case RPDB_rtrealms:
264*cf5a6c84SAndroid Build Coastguard Worker alist = rt_realms;
265*cf5a6c84SAndroid Build Coastguard Worker if (!rtrealms_init) {
266*cf5a6c84SAndroid Build Coastguard Worker rtrealms_init = 1;
267*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_realms,0,"unspec");
268*cf5a6c84SAndroid Build Coastguard Worker parseRPDB("/etc/iproute2/rt_realms", alist, ARRAY_LEN(rt_realms));
269*cf5a6c84SAndroid Build Coastguard Worker }
270*cf5a6c84SAndroid Build Coastguard Worker break;
271*cf5a6c84SAndroid Build Coastguard Worker case RPDB_rtscopes:
272*cf5a6c84SAndroid Build Coastguard Worker alist = rt_scope;
273*cf5a6c84SAndroid Build Coastguard Worker if (!rtscope_init) {
274*cf5a6c84SAndroid Build Coastguard Worker rtscope_init = 1;
275*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_scope,0,"global");
276*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_scope,200,"site");
277*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_scope,253,"link");
278*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_scope,254,"host");
279*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_scope,255,"nowhere");
280*cf5a6c84SAndroid Build Coastguard Worker parseRPDB("/etc/iproute2/rt_scopes", alist, ARRAY_LEN(rt_scope));
281*cf5a6c84SAndroid Build Coastguard Worker }
282*cf5a6c84SAndroid Build Coastguard Worker break;
283*cf5a6c84SAndroid Build Coastguard Worker case RPDB_rttables:
284*cf5a6c84SAndroid Build Coastguard Worker alist = rt_tables;
285*cf5a6c84SAndroid Build Coastguard Worker if (!rttable_init) {
286*cf5a6c84SAndroid Build Coastguard Worker rttable_init = 1;
287*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_tables,RT_TABLE_DEFAULT,"default");
288*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_tables,RT_TABLE_MAIN,"main");
289*cf5a6c84SAndroid Build Coastguard Worker init_arglist(rt_tables,RT_TABLE_LOCAL,"local");
290*cf5a6c84SAndroid Build Coastguard Worker parseRPDB("/etc/iproute2/rt_tables", alist, ARRAY_LEN(rt_tables));
291*cf5a6c84SAndroid Build Coastguard Worker }
292*cf5a6c84SAndroid Build Coastguard Worker break;
293*cf5a6c84SAndroid Build Coastguard Worker default:
294*cf5a6c84SAndroid Build Coastguard Worker error_exit("wrong database");
295*cf5a6c84SAndroid Build Coastguard Worker break; // Unreachable code.
296*cf5a6c84SAndroid Build Coastguard Worker }
297*cf5a6c84SAndroid Build Coastguard Worker return alist;
298*cf5a6c84SAndroid Build Coastguard Worker }
299*cf5a6c84SAndroid Build Coastguard Worker
300*cf5a6c84SAndroid Build Coastguard Worker /*
301*cf5a6c84SAndroid Build Coastguard Worker * Parse RPBD tables (if not parsed already).
302*cf5a6c84SAndroid Build Coastguard Worker * return RPDB table name as per idx.
303*cf5a6c84SAndroid Build Coastguard Worker */
namefromRPDB(int idx,u_int8_t whichDB)304*cf5a6c84SAndroid Build Coastguard Worker static char *namefromRPDB(int idx, u_int8_t whichDB)
305*cf5a6c84SAndroid Build Coastguard Worker {
306*cf5a6c84SAndroid Build Coastguard Worker struct arglist **alist;
307*cf5a6c84SAndroid Build Coastguard Worker
308*cf5a6c84SAndroid Build Coastguard Worker if (idx < 0 || idx >= RPDB_ENTRIES) {
309*cf5a6c84SAndroid Build Coastguard Worker snprintf(toybuf, RPDB_ENTRIES, "%u", idx);
310*cf5a6c84SAndroid Build Coastguard Worker return toybuf;
311*cf5a6c84SAndroid Build Coastguard Worker }
312*cf5a6c84SAndroid Build Coastguard Worker
313*cf5a6c84SAndroid Build Coastguard Worker alist = getlist(whichDB);
314*cf5a6c84SAndroid Build Coastguard Worker
315*cf5a6c84SAndroid Build Coastguard Worker if (alist[idx] && alist[idx]->name) return alist[idx]->name;
316*cf5a6c84SAndroid Build Coastguard Worker
317*cf5a6c84SAndroid Build Coastguard Worker if (whichDB == RPDB_rtdsfield) snprintf(toybuf, RPDB_ENTRIES, "0x%02x", idx);
318*cf5a6c84SAndroid Build Coastguard Worker else snprintf(toybuf, RPDB_ENTRIES, "%u", idx);
319*cf5a6c84SAndroid Build Coastguard Worker
320*cf5a6c84SAndroid Build Coastguard Worker return toybuf;
321*cf5a6c84SAndroid Build Coastguard Worker }
322*cf5a6c84SAndroid Build Coastguard Worker
idxfromRPDB(char * name,u_int8_t whichDB)323*cf5a6c84SAndroid Build Coastguard Worker static int idxfromRPDB(char *name, u_int8_t whichDB)
324*cf5a6c84SAndroid Build Coastguard Worker {
325*cf5a6c84SAndroid Build Coastguard Worker struct arglist **alist;
326*cf5a6c84SAndroid Build Coastguard Worker long i = 0;
327*cf5a6c84SAndroid Build Coastguard Worker char *ptr = NULL;
328*cf5a6c84SAndroid Build Coastguard Worker
329*cf5a6c84SAndroid Build Coastguard Worker for (alist = getlist(whichDB); i < RPDB_ENTRIES; i++) {
330*cf5a6c84SAndroid Build Coastguard Worker if (!alist[i] || !alist[i]->name) continue;
331*cf5a6c84SAndroid Build Coastguard Worker if (!strcmp(alist[i]->name, name)) return i;
332*cf5a6c84SAndroid Build Coastguard Worker }
333*cf5a6c84SAndroid Build Coastguard Worker i = strtol(name, &ptr, 0);
334*cf5a6c84SAndroid Build Coastguard Worker if (errno || (ptr && *ptr) || i < 0 || i > 255)
335*cf5a6c84SAndroid Build Coastguard Worker return -1;
336*cf5a6c84SAndroid Build Coastguard Worker return i;
337*cf5a6c84SAndroid Build Coastguard Worker }
338*cf5a6c84SAndroid Build Coastguard Worker
rtmtype_idx2str(u_int8_t idx)339*cf5a6c84SAndroid Build Coastguard Worker static char *rtmtype_idx2str(u_int8_t idx)
340*cf5a6c84SAndroid Build Coastguard Worker {
341*cf5a6c84SAndroid Build Coastguard Worker char *name = idx_to_string(idx, rtmtypes);
342*cf5a6c84SAndroid Build Coastguard Worker
343*cf5a6c84SAndroid Build Coastguard Worker if (!name) snprintf(toybuf, RPDB_ENTRIES, "%u", idx);
344*cf5a6c84SAndroid Build Coastguard Worker else snprintf(toybuf, sizeof(toybuf), "%s", name);
345*cf5a6c84SAndroid Build Coastguard Worker return toybuf;
346*cf5a6c84SAndroid Build Coastguard Worker }
347*cf5a6c84SAndroid Build Coastguard Worker
rtmtype_str2idx(char * name)348*cf5a6c84SAndroid Build Coastguard Worker static int rtmtype_str2idx(char *name)
349*cf5a6c84SAndroid Build Coastguard Worker {
350*cf5a6c84SAndroid Build Coastguard Worker int idx = string_to_idx(name, rtmtypes);
351*cf5a6c84SAndroid Build Coastguard Worker
352*cf5a6c84SAndroid Build Coastguard Worker if (idx < 0) return atolx_range(name, 0, 255);
353*cf5a6c84SAndroid Build Coastguard Worker return idx;
354*cf5a6c84SAndroid Build Coastguard Worker }
355*cf5a6c84SAndroid Build Coastguard Worker
356*cf5a6c84SAndroid Build Coastguard Worker /*
357*cf5a6c84SAndroid Build Coastguard Worker * Used to get the prefix value in binary form.
358*cf5a6c84SAndroid Build Coastguard Worker * For IPv4: non-standard parsing used; as 10.10 will be treated as 10.10.0.0
359*cf5a6c84SAndroid Build Coastguard Worker * unlike inet_aton which is 10.0.0.10
360*cf5a6c84SAndroid Build Coastguard Worker */
get_prefix(uint32_t * addr,uint8_t * af,char * name,int family)361*cf5a6c84SAndroid Build Coastguard Worker static int get_prefix(uint32_t *addr, uint8_t *af, char *name, int family)
362*cf5a6c84SAndroid Build Coastguard Worker {
363*cf5a6c84SAndroid Build Coastguard Worker if (family == AF_PACKET) error_exit("'%s' may be inet prefix", name);
364*cf5a6c84SAndroid Build Coastguard Worker if (!memcmp(name, "default", strlen(name))
365*cf5a6c84SAndroid Build Coastguard Worker || !memcmp(name, "all", strlen(name))
366*cf5a6c84SAndroid Build Coastguard Worker || !memcmp(name, "any", strlen(name))) {
367*cf5a6c84SAndroid Build Coastguard Worker *af = family;
368*cf5a6c84SAndroid Build Coastguard Worker return 0;
369*cf5a6c84SAndroid Build Coastguard Worker }
370*cf5a6c84SAndroid Build Coastguard Worker if (strchr(name, ':')) {
371*cf5a6c84SAndroid Build Coastguard Worker *af = AF_INET6;
372*cf5a6c84SAndroid Build Coastguard Worker if (family != AF_UNSPEC && family != AF_INET6) return 1;
373*cf5a6c84SAndroid Build Coastguard Worker if (inet_pton(AF_INET6, name, (void *)addr) != 1)
374*cf5a6c84SAndroid Build Coastguard Worker return 1;
375*cf5a6c84SAndroid Build Coastguard Worker } else { // for IPv4.
376*cf5a6c84SAndroid Build Coastguard Worker char *ptr = name;
377*cf5a6c84SAndroid Build Coastguard Worker uint8_t count = 0;
378*cf5a6c84SAndroid Build Coastguard Worker
379*cf5a6c84SAndroid Build Coastguard Worker *af = AF_INET;
380*cf5a6c84SAndroid Build Coastguard Worker if (family != AF_UNSPEC && family != AF_INET) return 1;
381*cf5a6c84SAndroid Build Coastguard Worker while (*ptr) {
382*cf5a6c84SAndroid Build Coastguard Worker int val, len = 0;
383*cf5a6c84SAndroid Build Coastguard Worker
384*cf5a6c84SAndroid Build Coastguard Worker if (*ptr == '.') ptr++;
385*cf5a6c84SAndroid Build Coastguard Worker sscanf(ptr, "%d%n", &val, &len);
386*cf5a6c84SAndroid Build Coastguard Worker if (!len || len > 3 || val < 0 || val > 255 || count > 3) return 1;
387*cf5a6c84SAndroid Build Coastguard Worker ptr += len;
388*cf5a6c84SAndroid Build Coastguard Worker ((uint8_t*)addr)[count++] = val;
389*cf5a6c84SAndroid Build Coastguard Worker }
390*cf5a6c84SAndroid Build Coastguard Worker }
391*cf5a6c84SAndroid Build Coastguard Worker return 0;
392*cf5a6c84SAndroid Build Coastguard Worker }
393*cf5a6c84SAndroid Build Coastguard Worker
394*cf5a6c84SAndroid Build Coastguard Worker /*
395*cf5a6c84SAndroid Build Coastguard Worker * Used to calculate netmask, which can be in the form of
396*cf5a6c84SAndroid Build Coastguard Worker * either 255.255.255.0 or 24 or default or any or all strings.
397*cf5a6c84SAndroid Build Coastguard Worker */
get_nmask_prefix(uint32_t * netmask,uint8_t af,char * name,uint8_t family)398*cf5a6c84SAndroid Build Coastguard Worker static int get_nmask_prefix(uint32_t *netmask, uint8_t af,
399*cf5a6c84SAndroid Build Coastguard Worker char *name, uint8_t family)
400*cf5a6c84SAndroid Build Coastguard Worker {
401*cf5a6c84SAndroid Build Coastguard Worker char *ptr;
402*cf5a6c84SAndroid Build Coastguard Worker uint32_t naddr[4] = {0,};
403*cf5a6c84SAndroid Build Coastguard Worker uint64_t plen;
404*cf5a6c84SAndroid Build Coastguard Worker uint8_t naf = AF_UNSPEC;
405*cf5a6c84SAndroid Build Coastguard Worker
406*cf5a6c84SAndroid Build Coastguard Worker *netmask = (af == AF_INET6) ? 128 : 32; // set default netmask
407*cf5a6c84SAndroid Build Coastguard Worker plen = strtoul(name, &ptr, 0);
408*cf5a6c84SAndroid Build Coastguard Worker
409*cf5a6c84SAndroid Build Coastguard Worker if (!ptr || ptr == name || *ptr || !plen || plen > *netmask) {
410*cf5a6c84SAndroid Build Coastguard Worker if (get_prefix(naddr, &naf, name, family)) return -1;
411*cf5a6c84SAndroid Build Coastguard Worker if (naf == AF_INET) {
412*cf5a6c84SAndroid Build Coastguard Worker uint32_t mask = htonl(*naddr), host = ~mask;
413*cf5a6c84SAndroid Build Coastguard Worker if (host & (host + 1)) return -1;
414*cf5a6c84SAndroid Build Coastguard Worker for (plen = 0; mask; mask <<= 1) ++plen;
415*cf5a6c84SAndroid Build Coastguard Worker if (plen > 32) return -1;
416*cf5a6c84SAndroid Build Coastguard Worker }
417*cf5a6c84SAndroid Build Coastguard Worker }
418*cf5a6c84SAndroid Build Coastguard Worker *netmask = plen;
419*cf5a6c84SAndroid Build Coastguard Worker return 0;
420*cf5a6c84SAndroid Build Coastguard Worker }
421*cf5a6c84SAndroid Build Coastguard Worker
422*cf5a6c84SAndroid Build Coastguard Worker /*
423*cf5a6c84SAndroid Build Coastguard Worker * Parse prefix, which will be in form of
424*cf5a6c84SAndroid Build Coastguard Worker * either default or default/default or default/24 or default/255.255.255.0
425*cf5a6c84SAndroid Build Coastguard Worker * or 10.20.30.40 or 10.20.30.40/default or 10.20.30.40/24
426*cf5a6c84SAndroid Build Coastguard Worker * or 10.20.30.40/255.255.255.0
427*cf5a6c84SAndroid Build Coastguard Worker */
parse_prefix(uint32_t * addr,uint32_t * netmask,uint8_t * len,char * name,int family)428*cf5a6c84SAndroid Build Coastguard Worker static void parse_prefix(uint32_t *addr, uint32_t *netmask, uint8_t *len,
429*cf5a6c84SAndroid Build Coastguard Worker char *name, int family)
430*cf5a6c84SAndroid Build Coastguard Worker {
431*cf5a6c84SAndroid Build Coastguard Worker uint8_t af = AF_UNSPEC;
432*cf5a6c84SAndroid Build Coastguard Worker char *slash = strchr(name, '/');
433*cf5a6c84SAndroid Build Coastguard Worker
434*cf5a6c84SAndroid Build Coastguard Worker if (slash) *slash = 0;
435*cf5a6c84SAndroid Build Coastguard Worker if (get_prefix(addr, &af, name, family)) error_exit("Invalid prefix");
436*cf5a6c84SAndroid Build Coastguard Worker
437*cf5a6c84SAndroid Build Coastguard Worker if (slash) { // grab netmask.
438*cf5a6c84SAndroid Build Coastguard Worker if (get_nmask_prefix(netmask, af, slash+1, family))
439*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid prefix");
440*cf5a6c84SAndroid Build Coastguard Worker *slash ='/';
441*cf5a6c84SAndroid Build Coastguard Worker }
442*cf5a6c84SAndroid Build Coastguard Worker else if (af == AF_INET && *addr) *netmask = 32;
443*cf5a6c84SAndroid Build Coastguard Worker else if (af == AF_INET6 && (*addr || *(addr+3))) *netmask = 128;
444*cf5a6c84SAndroid Build Coastguard Worker
445*cf5a6c84SAndroid Build Coastguard Worker if (!*addr && !slash && !af) *len = 0;
446*cf5a6c84SAndroid Build Coastguard Worker else *len = (af == AF_INET6) ? 16 : 4;
447*cf5a6c84SAndroid Build Coastguard Worker }
448*cf5a6c84SAndroid Build Coastguard Worker
449*cf5a6c84SAndroid Build Coastguard Worker /*
450*cf5a6c84SAndroid Build Coastguard Worker * Add a route attribute to a buffer; this is primarily used for extended
451*cf5a6c84SAndroid Build Coastguard Worker * attributes which get collected in a separate buffer from the normal route
452*cf5a6c84SAndroid Build Coastguard Worker * attributes and later get added to the main rtm message.
453*cf5a6c84SAndroid Build Coastguard Worker */
add_varlen_rtattr_to_buffer(struct rtattr * rta,int maxlen,int type,void * data,int alen)454*cf5a6c84SAndroid Build Coastguard Worker static void add_varlen_rtattr_to_buffer(struct rtattr *rta, int maxlen,
455*cf5a6c84SAndroid Build Coastguard Worker int type, void *data, int alen) {
456*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *subrta;
457*cf5a6c84SAndroid Build Coastguard Worker int len = RTA_LENGTH(alen);
458*cf5a6c84SAndroid Build Coastguard Worker if (RTA_ALIGN(rta->rta_len) + RTA_ALIGN(len) > maxlen) {
459*cf5a6c84SAndroid Build Coastguard Worker error_exit("RTA exceeds max length %d", maxlen);
460*cf5a6c84SAndroid Build Coastguard Worker }
461*cf5a6c84SAndroid Build Coastguard Worker subrta = (struct rtattr*)(((char*)rta) + RTA_ALIGN(rta->rta_len));
462*cf5a6c84SAndroid Build Coastguard Worker subrta->rta_type = type;
463*cf5a6c84SAndroid Build Coastguard Worker subrta->rta_len = len;
464*cf5a6c84SAndroid Build Coastguard Worker if (alen) {
465*cf5a6c84SAndroid Build Coastguard Worker memcpy(RTA_DATA(subrta), data, alen);
466*cf5a6c84SAndroid Build Coastguard Worker }
467*cf5a6c84SAndroid Build Coastguard Worker rta->rta_len = NLMSG_ALIGN(rta->rta_len) + RTA_ALIGN(len);
468*cf5a6c84SAndroid Build Coastguard Worker }
469*cf5a6c84SAndroid Build Coastguard Worker
add_uint32_rtattr_to_buffer(struct rtattr * rta,int maxlen,int type,uint32_t attr)470*cf5a6c84SAndroid Build Coastguard Worker static void add_uint32_rtattr_to_buffer(struct rtattr *rta, int maxlen,
471*cf5a6c84SAndroid Build Coastguard Worker int type, uint32_t attr) {
472*cf5a6c84SAndroid Build Coastguard Worker add_varlen_rtattr_to_buffer(rta, maxlen, type, (char*)&attr, sizeof(attr));
473*cf5a6c84SAndroid Build Coastguard Worker }
474*cf5a6c84SAndroid Build Coastguard Worker
475*cf5a6c84SAndroid Build Coastguard Worker /*
476*cf5a6c84SAndroid Build Coastguard Worker * Add a route attribute to a RTM message.
477*cf5a6c84SAndroid Build Coastguard Worker */
add_string_to_rtattr(struct nlmsghdr * n,int maxlen,int type,void * data,int alen)478*cf5a6c84SAndroid Build Coastguard Worker static void add_string_to_rtattr(struct nlmsghdr *n, int maxlen,
479*cf5a6c84SAndroid Build Coastguard Worker int type, void *data, int alen)
480*cf5a6c84SAndroid Build Coastguard Worker {
481*cf5a6c84SAndroid Build Coastguard Worker int len = RTA_LENGTH(alen);
482*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *rta;
483*cf5a6c84SAndroid Build Coastguard Worker
484*cf5a6c84SAndroid Build Coastguard Worker if ((int)(NLMSG_ALIGN(n->nlmsg_len) + len) > maxlen) return;
485*cf5a6c84SAndroid Build Coastguard Worker rta = (struct rtattr*)(((char*)n) + NLMSG_ALIGN(n->nlmsg_len));
486*cf5a6c84SAndroid Build Coastguard Worker rta->rta_type = type;
487*cf5a6c84SAndroid Build Coastguard Worker rta->rta_len = len;
488*cf5a6c84SAndroid Build Coastguard Worker memcpy(RTA_DATA(rta), data, alen);
489*cf5a6c84SAndroid Build Coastguard Worker n->nlmsg_len = NLMSG_ALIGN(n->nlmsg_len) + len;
490*cf5a6c84SAndroid Build Coastguard Worker }
491*cf5a6c84SAndroid Build Coastguard Worker
492*cf5a6c84SAndroid Build Coastguard Worker
493*cf5a6c84SAndroid Build Coastguard Worker
494*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
495*cf5a6c84SAndroid Build Coastguard Worker // Code for ip link.
496*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
497*cf5a6c84SAndroid Build Coastguard Worker #ifndef NLMSG_TAIL
498*cf5a6c84SAndroid Build Coastguard Worker #define NLMSG_TAIL(nmsg) \
499*cf5a6c84SAndroid Build Coastguard Worker ((struct rtattr *) (((void *) (nmsg)) + NLMSG_ALIGN((nmsg)->nlmsg_len)))
500*cf5a6c84SAndroid Build Coastguard Worker #endif
501*cf5a6c84SAndroid Build Coastguard Worker
get_ifaceindex(char * name,int ext)502*cf5a6c84SAndroid Build Coastguard Worker static uint32_t get_ifaceindex(char *name, int ext)
503*cf5a6c84SAndroid Build Coastguard Worker {
504*cf5a6c84SAndroid Build Coastguard Worker struct if_nameindex *if_ni, *i;
505*cf5a6c84SAndroid Build Coastguard Worker int index = -1;
506*cf5a6c84SAndroid Build Coastguard Worker
507*cf5a6c84SAndroid Build Coastguard Worker if_ni = if_nameindex();
508*cf5a6c84SAndroid Build Coastguard Worker if (!if_ni) perror_exit("if_nameindex");
509*cf5a6c84SAndroid Build Coastguard Worker
510*cf5a6c84SAndroid Build Coastguard Worker for (i = if_ni; i->if_index && i->if_name; i++)
511*cf5a6c84SAndroid Build Coastguard Worker if (!strcmp(name, i->if_name)) {
512*cf5a6c84SAndroid Build Coastguard Worker index = i->if_index;
513*cf5a6c84SAndroid Build Coastguard Worker break;
514*cf5a6c84SAndroid Build Coastguard Worker }
515*cf5a6c84SAndroid Build Coastguard Worker if_freenameindex(if_ni);
516*cf5a6c84SAndroid Build Coastguard Worker if (index == -1 && ext) perror_exit("can't find device '%s'", name);
517*cf5a6c84SAndroid Build Coastguard Worker return index;
518*cf5a6c84SAndroid Build Coastguard Worker }
519*cf5a6c84SAndroid Build Coastguard Worker
fill_hwaddr(char * arg,int len,unsigned char * address)520*cf5a6c84SAndroid Build Coastguard Worker static void fill_hwaddr(char *arg, int len, unsigned char *address)
521*cf5a6c84SAndroid Build Coastguard Worker {
522*cf5a6c84SAndroid Build Coastguard Worker int count = 0, val, length;
523*cf5a6c84SAndroid Build Coastguard Worker
524*cf5a6c84SAndroid Build Coastguard Worker while (count < len) {
525*cf5a6c84SAndroid Build Coastguard Worker val = length = 0;
526*cf5a6c84SAndroid Build Coastguard Worker if (!arg) error_exit("bad hw-addr '%s'", "");
527*cf5a6c84SAndroid Build Coastguard Worker if (*arg == ':') arg++, count++;
528*cf5a6c84SAndroid Build Coastguard Worker sscanf(arg, "%2x%n", &val, &length);
529*cf5a6c84SAndroid Build Coastguard Worker if (!length || length > 2)
530*cf5a6c84SAndroid Build Coastguard Worker error_exit("bad hw-addr '%s'", arg);
531*cf5a6c84SAndroid Build Coastguard Worker arg += length;
532*cf5a6c84SAndroid Build Coastguard Worker count += length;
533*cf5a6c84SAndroid Build Coastguard Worker *address++ = val;
534*cf5a6c84SAndroid Build Coastguard Worker }
535*cf5a6c84SAndroid Build Coastguard Worker }
536*cf5a6c84SAndroid Build Coastguard Worker
537*cf5a6c84SAndroid Build Coastguard Worker // Multimach = 1, single match = 0
get_flag_string(struct arglist * aflags,int flags,int ismulti)538*cf5a6c84SAndroid Build Coastguard Worker static char *get_flag_string(struct arglist *aflags, int flags, int ismulti)
539*cf5a6c84SAndroid Build Coastguard Worker {
540*cf5a6c84SAndroid Build Coastguard Worker struct arglist *p = aflags;
541*cf5a6c84SAndroid Build Coastguard Worker char *out = NULL, *tmp = NULL;
542*cf5a6c84SAndroid Build Coastguard Worker
543*cf5a6c84SAndroid Build Coastguard Worker for (; p->name; p++) {
544*cf5a6c84SAndroid Build Coastguard Worker int test = (ismulti ? p->idx & flags : 0) || p->idx == flags;
545*cf5a6c84SAndroid Build Coastguard Worker if (test) { // flags can be zero
546*cf5a6c84SAndroid Build Coastguard Worker tmp = out ? xmprintf("%s,%s", out, p->name) : xmprintf("%s", p->name);
547*cf5a6c84SAndroid Build Coastguard Worker if (out) free(out);
548*cf5a6c84SAndroid Build Coastguard Worker out = tmp;
549*cf5a6c84SAndroid Build Coastguard Worker }
550*cf5a6c84SAndroid Build Coastguard Worker }
551*cf5a6c84SAndroid Build Coastguard Worker return out;
552*cf5a6c84SAndroid Build Coastguard Worker }
553*cf5a6c84SAndroid Build Coastguard Worker
vlan_parse_opt(char ** argv,struct nlmsghdr * n,unsigned int size)554*cf5a6c84SAndroid Build Coastguard Worker static void vlan_parse_opt(char **argv, struct nlmsghdr *n, unsigned int size)
555*cf5a6c84SAndroid Build Coastguard Worker {
556*cf5a6c84SAndroid Build Coastguard Worker struct arglist vlan_optlist[] = {{"id", 0}, {"protocol", 1},
557*cf5a6c84SAndroid Build Coastguard Worker {"reorder_hdr", 2}, {"gvrp", 3}, {NULL,-1}};
558*cf5a6c84SAndroid Build Coastguard Worker struct arglist vlan_protolist[] = {{"802.1q", 0}, {"802.1ad", 1}, {NULL,-1}};
559*cf5a6c84SAndroid Build Coastguard Worker struct arglist on_off[] = { {"on", 0}, {"off", 1}, {NULL,-1}};
560*cf5a6c84SAndroid Build Coastguard Worker int idx;
561*cf5a6c84SAndroid Build Coastguard Worker struct ifla_vlan_flags flags;
562*cf5a6c84SAndroid Build Coastguard Worker
563*cf5a6c84SAndroid Build Coastguard Worker memset(&flags, 0, sizeof(flags));
564*cf5a6c84SAndroid Build Coastguard Worker for (; *argv; argv++) {
565*cf5a6c84SAndroid Build Coastguard Worker int param, proto;
566*cf5a6c84SAndroid Build Coastguard Worker
567*cf5a6c84SAndroid Build Coastguard Worker if ((idx = substring_to_idx(*argv++, vlan_optlist)) == -1) help_exit(0);
568*cf5a6c84SAndroid Build Coastguard Worker switch (idx) {
569*cf5a6c84SAndroid Build Coastguard Worker case 0: // ARG_id
570*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) help_exit(0);
571*cf5a6c84SAndroid Build Coastguard Worker param = atolx(*argv);
572*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(n, size, IFLA_VLAN_ID, ¶m, sizeof(param));
573*cf5a6c84SAndroid Build Coastguard Worker break;
574*cf5a6c84SAndroid Build Coastguard Worker case 1: // ARG_protocol
575*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) error_exit("Invalid vlan id.");
576*cf5a6c84SAndroid Build Coastguard Worker if ((idx = substring_to_idx(*argv, vlan_protolist)) == -1) help_exit(0);
577*cf5a6c84SAndroid Build Coastguard Worker if (!idx) proto = ETH_P_8021Q; // PROTO_8021Q - 0
578*cf5a6c84SAndroid Build Coastguard Worker else if (idx == 1) proto = 0x88A8; // ETH Protocol - 8021AD
579*cf5a6c84SAndroid Build Coastguard Worker // IFLA VLAN PROTOCOL - 5
580*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(n, size, 5, &proto, sizeof(proto));
581*cf5a6c84SAndroid Build Coastguard Worker break;
582*cf5a6c84SAndroid Build Coastguard Worker case 2: // ARG_reorder_hdr
583*cf5a6c84SAndroid Build Coastguard Worker case 3: // ARG_gvrp
584*cf5a6c84SAndroid Build Coastguard Worker if ((param = substring_to_idx(*argv, on_off)) == -1) help_exit(0);
585*cf5a6c84SAndroid Build Coastguard Worker
586*cf5a6c84SAndroid Build Coastguard Worker flags.mask |= (idx -1); // VLAN FLAG REORDER Header
587*cf5a6c84SAndroid Build Coastguard Worker flags.flags &= ~(idx -1); // VLAN FLAG REORDER Header
588*cf5a6c84SAndroid Build Coastguard Worker if (!param) flags.flags |= (idx -1); // VLAN FLAG REORDER Header
589*cf5a6c84SAndroid Build Coastguard Worker break;
590*cf5a6c84SAndroid Build Coastguard Worker }
591*cf5a6c84SAndroid Build Coastguard Worker }
592*cf5a6c84SAndroid Build Coastguard Worker if (flags.mask)
593*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(n, size, IFLA_VLAN_FLAGS, &flags, sizeof(flags));
594*cf5a6c84SAndroid Build Coastguard Worker }
595*cf5a6c84SAndroid Build Coastguard Worker
linkupdate(char ** argv)596*cf5a6c84SAndroid Build Coastguard Worker static int linkupdate(char **argv)
597*cf5a6c84SAndroid Build Coastguard Worker {
598*cf5a6c84SAndroid Build Coastguard Worker struct {
599*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr mhdr;
600*cf5a6c84SAndroid Build Coastguard Worker struct ifinfomsg info;
601*cf5a6c84SAndroid Build Coastguard Worker char buf[1024];
602*cf5a6c84SAndroid Build Coastguard Worker } request;
603*cf5a6c84SAndroid Build Coastguard Worker char *name, *dev, *type, *link, *addr;
604*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *attr = NULL;
605*cf5a6c84SAndroid Build Coastguard Worker int len = 0, add = (*argv[-1] == 'a') ? 1 : 0;
606*cf5a6c84SAndroid Build Coastguard Worker
607*cf5a6c84SAndroid Build Coastguard Worker name = dev = type = link = addr = NULL;
608*cf5a6c84SAndroid Build Coastguard Worker for (; *argv; argv++) {
609*cf5a6c84SAndroid Build Coastguard Worker struct arglist objectlist[] = { {"type", 0}, {"name", 1}, {"link", 2},
610*cf5a6c84SAndroid Build Coastguard Worker {"address", 3}, {NULL,-1}};
611*cf5a6c84SAndroid Build Coastguard Worker uint8_t idx = substring_to_idx(*argv, objectlist);
612*cf5a6c84SAndroid Build Coastguard Worker
613*cf5a6c84SAndroid Build Coastguard Worker if (!idx) {
614*cf5a6c84SAndroid Build Coastguard Worker type = *++argv;
615*cf5a6c84SAndroid Build Coastguard Worker break;
616*cf5a6c84SAndroid Build Coastguard Worker }
617*cf5a6c84SAndroid Build Coastguard Worker else if (idx == 1) dev = name = *++argv;
618*cf5a6c84SAndroid Build Coastguard Worker else if (idx == 2) link = *++argv;
619*cf5a6c84SAndroid Build Coastguard Worker else if (idx == 3) addr = *++argv;
620*cf5a6c84SAndroid Build Coastguard Worker else if (!dev) name = dev = *argv;
621*cf5a6c84SAndroid Build Coastguard Worker }
622*cf5a6c84SAndroid Build Coastguard Worker
623*cf5a6c84SAndroid Build Coastguard Worker if (!name && !add)
624*cf5a6c84SAndroid Build Coastguard Worker error_exit("Not enough information: \"dev\" argument is required.\n");
625*cf5a6c84SAndroid Build Coastguard Worker else if (!type && add)
626*cf5a6c84SAndroid Build Coastguard Worker error_exit("Not enough information: \"type\" argument is required.\n");
627*cf5a6c84SAndroid Build Coastguard Worker
628*cf5a6c84SAndroid Build Coastguard Worker memset(&request, 0, sizeof(request));
629*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_len = NLMSG_LENGTH(sizeof(struct ifinfomsg));
630*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_flags = NLM_F_REQUEST|NLM_F_ACK;
631*cf5a6c84SAndroid Build Coastguard Worker if (add) {
632*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_flags |= NLM_F_CREATE|NLM_F_EXCL;
633*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_type = RTM_NEWLINK;
634*cf5a6c84SAndroid Build Coastguard Worker } else {
635*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_type = RTM_DELLINK;
636*cf5a6c84SAndroid Build Coastguard Worker request.info.ifi_index = get_ifaceindex(name, 1);
637*cf5a6c84SAndroid Build Coastguard Worker }
638*cf5a6c84SAndroid Build Coastguard Worker request.info.ifi_family = AF_UNSPEC;
639*cf5a6c84SAndroid Build Coastguard Worker attr = NLMSG_TAIL(&request.mhdr);
640*cf5a6c84SAndroid Build Coastguard Worker if (type) {
641*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
642*cf5a6c84SAndroid Build Coastguard Worker IFLA_LINKINFO, NULL, 0);
643*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
644*cf5a6c84SAndroid Build Coastguard Worker IFLA_INFO_KIND, type, strlen(type));
645*cf5a6c84SAndroid Build Coastguard Worker if (!strcmp(type, "vlan")) {
646*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *data = NLMSG_TAIL(&request.mhdr);
647*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
648*cf5a6c84SAndroid Build Coastguard Worker IFLA_INFO_DATA, NULL, 0);
649*cf5a6c84SAndroid Build Coastguard Worker vlan_parse_opt(++argv, &request.mhdr, sizeof(request));
650*cf5a6c84SAndroid Build Coastguard Worker data->rta_len = (void *)NLMSG_TAIL(&request.mhdr) - (void *)data;
651*cf5a6c84SAndroid Build Coastguard Worker }
652*cf5a6c84SAndroid Build Coastguard Worker attr->rta_len = (void *)NLMSG_TAIL(&request.mhdr) - (void *)attr;
653*cf5a6c84SAndroid Build Coastguard Worker }
654*cf5a6c84SAndroid Build Coastguard Worker
655*cf5a6c84SAndroid Build Coastguard Worker if (link) {
656*cf5a6c84SAndroid Build Coastguard Worker uint32_t idx = get_ifaceindex(link, 1);
657*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
658*cf5a6c84SAndroid Build Coastguard Worker IFLA_LINK, &idx, sizeof(uint32_t));
659*cf5a6c84SAndroid Build Coastguard Worker }
660*cf5a6c84SAndroid Build Coastguard Worker if (addr) {
661*cf5a6c84SAndroid Build Coastguard Worker char abuf[IF_NAMESIZE] = {0,};
662*cf5a6c84SAndroid Build Coastguard Worker
663*cf5a6c84SAndroid Build Coastguard Worker fill_hwaddr(addr, IF_NAMESIZE, (unsigned char *)abuf);
664*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
665*cf5a6c84SAndroid Build Coastguard Worker IFLA_ADDRESS, abuf, strlen(abuf));
666*cf5a6c84SAndroid Build Coastguard Worker }
667*cf5a6c84SAndroid Build Coastguard Worker if (!name) {
668*cf5a6c84SAndroid Build Coastguard Worker snprintf(toybuf, IFNAMSIZ, "%s%d", type, 0);
669*cf5a6c84SAndroid Build Coastguard Worker for (len = 1; ; len++) {
670*cf5a6c84SAndroid Build Coastguard Worker if (!get_ifaceindex(toybuf, 0)) break;
671*cf5a6c84SAndroid Build Coastguard Worker snprintf(toybuf, IFNAMSIZ, "%s%d", type, len);
672*cf5a6c84SAndroid Build Coastguard Worker }
673*cf5a6c84SAndroid Build Coastguard Worker name = toybuf;
674*cf5a6c84SAndroid Build Coastguard Worker }
675*cf5a6c84SAndroid Build Coastguard Worker len = strlen(name) + 1;
676*cf5a6c84SAndroid Build Coastguard Worker if (len < 2 || len > IFNAMSIZ) error_exit("Invalid device name.");
677*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request), IFLA_IFNAME, name, len);
678*cf5a6c84SAndroid Build Coastguard Worker
679*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(0, 0, 0, (void *)&request, request.mhdr.nlmsg_len);
680*cf5a6c84SAndroid Build Coastguard Worker return (filter_nlmesg(NULL,NULL));
681*cf5a6c84SAndroid Build Coastguard Worker }
682*cf5a6c84SAndroid Build Coastguard Worker
link_set(char ** argv)683*cf5a6c84SAndroid Build Coastguard Worker static int link_set(char **argv)
684*cf5a6c84SAndroid Build Coastguard Worker {
685*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist[] = {{"up", 0}, {"down", 1}, {"arp", 2},
686*cf5a6c84SAndroid Build Coastguard Worker {"multicast", 3}, {"dynamic", 4}, {"name", 5}, {"txqueuelen", 6},
687*cf5a6c84SAndroid Build Coastguard Worker {"mtu", 7},{"address", 8}, {"broadcast", 9}, {"dev", 10}, {NULL,-1}};
688*cf5a6c84SAndroid Build Coastguard Worker int case_flags[] = {IFF_NOARP,IFF_MULTICAST,IFF_DYNAMIC};
689*cf5a6c84SAndroid Build Coastguard Worker struct ifreq req;
690*cf5a6c84SAndroid Build Coastguard Worker int idx, flags = 0, masks = 0xffff, fd;
691*cf5a6c84SAndroid Build Coastguard Worker char *dev = NULL, *newdev = NULL, *newaddr = NULL, *newbrd = NULL;
692*cf5a6c84SAndroid Build Coastguard Worker int mtu = -1, txqlen = -1;
693*cf5a6c84SAndroid Build Coastguard Worker
694*cf5a6c84SAndroid Build Coastguard Worker int i = 0;
695*cf5a6c84SAndroid Build Coastguard Worker while (*argv) {
696*cf5a6c84SAndroid Build Coastguard Worker switch(idx = substring_to_idx(*argv, cmd_objectlist)) {
697*cf5a6c84SAndroid Build Coastguard Worker case 0:
698*cf5a6c84SAndroid Build Coastguard Worker flags |= IFF_UP; break;
699*cf5a6c84SAndroid Build Coastguard Worker case 1:
700*cf5a6c84SAndroid Build Coastguard Worker masks &= ~IFF_UP; break;
701*cf5a6c84SAndroid Build Coastguard Worker case 2:
702*cf5a6c84SAndroid Build Coastguard Worker case 3:
703*cf5a6c84SAndroid Build Coastguard Worker case 4:
704*cf5a6c84SAndroid Build Coastguard Worker ++argv;
705*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) help_exit(0);
706*cf5a6c84SAndroid Build Coastguard Worker else if (!strcmp(*argv, "on")) {
707*cf5a6c84SAndroid Build Coastguard Worker if (idx == 2) {
708*cf5a6c84SAndroid Build Coastguard Worker masks &= ~case_flags[idx-2];
709*cf5a6c84SAndroid Build Coastguard Worker flags &= ~case_flags[idx-2];
710*cf5a6c84SAndroid Build Coastguard Worker } else flags |= case_flags[idx-2];
711*cf5a6c84SAndroid Build Coastguard Worker } else if (!strcmp(*argv,"off")) {
712*cf5a6c84SAndroid Build Coastguard Worker if (idx == 2) {
713*cf5a6c84SAndroid Build Coastguard Worker masks |= case_flags[idx-2];
714*cf5a6c84SAndroid Build Coastguard Worker flags |= case_flags[idx-2];
715*cf5a6c84SAndroid Build Coastguard Worker } else masks &= ~case_flags[idx-2];
716*cf5a6c84SAndroid Build Coastguard Worker } else help_exit(0);
717*cf5a6c84SAndroid Build Coastguard Worker break;
718*cf5a6c84SAndroid Build Coastguard Worker case 5:
719*cf5a6c84SAndroid Build Coastguard Worker ++argv;
720*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) error_exit("Incomplete Command line");
721*cf5a6c84SAndroid Build Coastguard Worker newdev = *argv;
722*cf5a6c84SAndroid Build Coastguard Worker break;
723*cf5a6c84SAndroid Build Coastguard Worker case 6:
724*cf5a6c84SAndroid Build Coastguard Worker ++argv;
725*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) error_exit("Incomplete Command line");
726*cf5a6c84SAndroid Build Coastguard Worker txqlen = atolx(*argv);
727*cf5a6c84SAndroid Build Coastguard Worker break;
728*cf5a6c84SAndroid Build Coastguard Worker case 7:
729*cf5a6c84SAndroid Build Coastguard Worker ++argv;
730*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) error_exit("Incomplete Command line");
731*cf5a6c84SAndroid Build Coastguard Worker mtu = atolx(*argv);
732*cf5a6c84SAndroid Build Coastguard Worker break;
733*cf5a6c84SAndroid Build Coastguard Worker case 8:
734*cf5a6c84SAndroid Build Coastguard Worker ++argv;
735*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) error_exit("Incomplete Command line");
736*cf5a6c84SAndroid Build Coastguard Worker newaddr = *argv;
737*cf5a6c84SAndroid Build Coastguard Worker break;
738*cf5a6c84SAndroid Build Coastguard Worker case 9:
739*cf5a6c84SAndroid Build Coastguard Worker ++argv;
740*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) error_exit("Incomplete Command line");
741*cf5a6c84SAndroid Build Coastguard Worker newbrd = *argv;
742*cf5a6c84SAndroid Build Coastguard Worker break;
743*cf5a6c84SAndroid Build Coastguard Worker case 10:
744*cf5a6c84SAndroid Build Coastguard Worker ++argv;
745*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) error_exit("Incomplete Command line");
746*cf5a6c84SAndroid Build Coastguard Worker
747*cf5a6c84SAndroid Build Coastguard Worker default:
748*cf5a6c84SAndroid Build Coastguard Worker if (dev)
749*cf5a6c84SAndroid Build Coastguard Worker error_exit("Either \"dev\" is duplicate or %s is garbage",
750*cf5a6c84SAndroid Build Coastguard Worker *argv);
751*cf5a6c84SAndroid Build Coastguard Worker dev = *argv;
752*cf5a6c84SAndroid Build Coastguard Worker break;
753*cf5a6c84SAndroid Build Coastguard Worker }
754*cf5a6c84SAndroid Build Coastguard Worker ++argv;
755*cf5a6c84SAndroid Build Coastguard Worker }
756*cf5a6c84SAndroid Build Coastguard Worker
757*cf5a6c84SAndroid Build Coastguard Worker memset(&req, 0, sizeof(req));
758*cf5a6c84SAndroid Build Coastguard Worker if (!dev) error_exit("\"dev\" missing");
759*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(req.ifr_name, dev, IF_NAMESIZE);
760*cf5a6c84SAndroid Build Coastguard Worker fd = xsocket(AF_INET, SOCK_DGRAM, 0);
761*cf5a6c84SAndroid Build Coastguard Worker
762*cf5a6c84SAndroid Build Coastguard Worker if (newdev) {
763*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(req.ifr_ifru.ifru_newname, newdev, IF_NAMESIZE);
764*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCSIFNAME, &req);
765*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(req.ifr_name, newdev, IF_NAMESIZE);
766*cf5a6c84SAndroid Build Coastguard Worker }
767*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCGIFINDEX, &req);
768*cf5a6c84SAndroid Build Coastguard Worker
769*cf5a6c84SAndroid Build Coastguard Worker if (txqlen != -1) {
770*cf5a6c84SAndroid Build Coastguard Worker req.ifr_ifru.ifru_ivalue = txqlen;
771*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCSIFTXQLEN, &req);
772*cf5a6c84SAndroid Build Coastguard Worker }
773*cf5a6c84SAndroid Build Coastguard Worker
774*cf5a6c84SAndroid Build Coastguard Worker if (mtu != -1) {
775*cf5a6c84SAndroid Build Coastguard Worker req.ifr_ifru.ifru_mtu = mtu;
776*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCSIFMTU, &req);
777*cf5a6c84SAndroid Build Coastguard Worker }
778*cf5a6c84SAndroid Build Coastguard Worker
779*cf5a6c84SAndroid Build Coastguard Worker if (newaddr) {
780*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCGIFHWADDR, &req);
781*cf5a6c84SAndroid Build Coastguard Worker fill_hwaddr(newaddr, IF_NAMESIZE,
782*cf5a6c84SAndroid Build Coastguard Worker (unsigned char *)(req.ifr_hwaddr.sa_data));
783*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCSIFHWADDR, &req);
784*cf5a6c84SAndroid Build Coastguard Worker }
785*cf5a6c84SAndroid Build Coastguard Worker
786*cf5a6c84SAndroid Build Coastguard Worker if (newbrd) {
787*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCGIFHWADDR, &req);
788*cf5a6c84SAndroid Build Coastguard Worker fill_hwaddr(newbrd, IF_NAMESIZE,
789*cf5a6c84SAndroid Build Coastguard Worker (unsigned char *)(req.ifr_hwaddr.sa_data));
790*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCSIFHWBROADCAST, &req);
791*cf5a6c84SAndroid Build Coastguard Worker }
792*cf5a6c84SAndroid Build Coastguard Worker
793*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCGIFFLAGS, &req);
794*cf5a6c84SAndroid Build Coastguard Worker req.ifr_ifru.ifru_flags |= flags;
795*cf5a6c84SAndroid Build Coastguard Worker req.ifr_ifru.ifru_flags &= masks;
796*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCSIFFLAGS, &req);
797*cf5a6c84SAndroid Build Coastguard Worker xclose(fd);
798*cf5a6c84SAndroid Build Coastguard Worker return 0;
799*cf5a6c84SAndroid Build Coastguard Worker }
800*cf5a6c84SAndroid Build Coastguard Worker
print_stats(struct rtnl_link_stats * rtstat)801*cf5a6c84SAndroid Build Coastguard Worker static void print_stats(struct rtnl_link_stats *rtstat)
802*cf5a6c84SAndroid Build Coastguard Worker {
803*cf5a6c84SAndroid Build Coastguard Worker char *line_feed = (!TT.singleline ? "\n " : " ");
804*cf5a6c84SAndroid Build Coastguard Worker
805*cf5a6c84SAndroid Build Coastguard Worker if (TT.stats > 0) {
806*cf5a6c84SAndroid Build Coastguard Worker xprintf(" RX: bytes packets errors "
807*cf5a6c84SAndroid Build Coastguard Worker "dropped overrun mcast%s%-10u %-8u %-7u %-8u %-8u %-8u\n",
808*cf5a6c84SAndroid Build Coastguard Worker line_feed, rtstat->rx_bytes, rtstat->rx_packets, rtstat->rx_errors,
809*cf5a6c84SAndroid Build Coastguard Worker rtstat->rx_dropped, rtstat->rx_over_errors, rtstat->multicast);
810*cf5a6c84SAndroid Build Coastguard Worker if (TT.stats > 1) {
811*cf5a6c84SAndroid Build Coastguard Worker xprintf(" RX: errors length crc "
812*cf5a6c84SAndroid Build Coastguard Worker "frame fifo missed%s%-10u %-8u %-7u %-8u %-8u %-8u\n",
813*cf5a6c84SAndroid Build Coastguard Worker line_feed, rtstat->rx_errors, rtstat->rx_length_errors,
814*cf5a6c84SAndroid Build Coastguard Worker rtstat->rx_crc_errors, rtstat->rx_frame_errors,
815*cf5a6c84SAndroid Build Coastguard Worker rtstat->rx_fifo_errors, rtstat->rx_missed_errors);
816*cf5a6c84SAndroid Build Coastguard Worker }
817*cf5a6c84SAndroid Build Coastguard Worker xprintf(" TX: bytes packets errors "
818*cf5a6c84SAndroid Build Coastguard Worker "dropped carrier collsns%s%-10u %-8u %-7u %-8u %-8u %-8u\n",
819*cf5a6c84SAndroid Build Coastguard Worker line_feed, rtstat->tx_bytes, rtstat->tx_packets, rtstat->tx_errors,
820*cf5a6c84SAndroid Build Coastguard Worker rtstat->tx_dropped, rtstat->tx_carrier_errors, rtstat->collisions);
821*cf5a6c84SAndroid Build Coastguard Worker if (TT.stats > 1) {
822*cf5a6c84SAndroid Build Coastguard Worker xprintf(" TX: errors aborted fifo window "
823*cf5a6c84SAndroid Build Coastguard Worker "heartbeat%s%-10u %-8u %-7u %-8u %-8u\n",
824*cf5a6c84SAndroid Build Coastguard Worker line_feed, rtstat->tx_errors, rtstat->tx_aborted_errors,
825*cf5a6c84SAndroid Build Coastguard Worker rtstat->tx_fifo_errors, rtstat->tx_window_errors,
826*cf5a6c84SAndroid Build Coastguard Worker rtstat->tx_heartbeat_errors);
827*cf5a6c84SAndroid Build Coastguard Worker }
828*cf5a6c84SAndroid Build Coastguard Worker }
829*cf5a6c84SAndroid Build Coastguard Worker }
830*cf5a6c84SAndroid Build Coastguard Worker
print_link_output(struct linkdata * link)831*cf5a6c84SAndroid Build Coastguard Worker static int print_link_output(struct linkdata *link)
832*cf5a6c84SAndroid Build Coastguard Worker {
833*cf5a6c84SAndroid Build Coastguard Worker char *line_feed = " ", *flags,*peer = "brd";
834*cf5a6c84SAndroid Build Coastguard Worker struct arglist iface_flags[] = {{"",0},{"UP", IFF_UP},
835*cf5a6c84SAndroid Build Coastguard Worker {"BROADCAST", IFF_BROADCAST}, {"DEBUG", IFF_DEBUG},
836*cf5a6c84SAndroid Build Coastguard Worker {"LOOPBACK", IFF_LOOPBACK}, {"POINTOPOINT", IFF_POINTOPOINT},
837*cf5a6c84SAndroid Build Coastguard Worker {"NOTRAILERS", IFF_NOTRAILERS}, {"RUNNING", IFF_RUNNING},
838*cf5a6c84SAndroid Build Coastguard Worker {"NOARP", IFF_NOARP}, {"PROMISC",IFF_PROMISC},
839*cf5a6c84SAndroid Build Coastguard Worker {"ALLMULTI", IFF_ALLMULTI}, {"MASTER", IFF_MASTER}, {"SLAVE", IFF_SLAVE},
840*cf5a6c84SAndroid Build Coastguard Worker {"MULTICAST", IFF_MULTICAST}, {"PORTSEL", IFF_PORTSEL},
841*cf5a6c84SAndroid Build Coastguard Worker {"AUTOMEDIA", IFF_AUTOMEDIA}, {"DYNAMIC", IFF_DYNAMIC},
842*cf5a6c84SAndroid Build Coastguard Worker {"LOWER_UP", IFF_LOWER_UP}, {"DORMANT", IFF_DORMANT},
843*cf5a6c84SAndroid Build Coastguard Worker {"ECHO", IFF_ECHO}, {NULL,-1}};
844*cf5a6c84SAndroid Build Coastguard Worker
845*cf5a6c84SAndroid Build Coastguard Worker if (link->parent != -1) {
846*cf5a6c84SAndroid Build Coastguard Worker int fd = 0;
847*cf5a6c84SAndroid Build Coastguard Worker struct ifreq req;
848*cf5a6c84SAndroid Build Coastguard Worker
849*cf5a6c84SAndroid Build Coastguard Worker memset(&req, 0, sizeof(req));
850*cf5a6c84SAndroid Build Coastguard Worker if_indextoname( link->parent,req.ifr_ifrn.ifrn_name);
851*cf5a6c84SAndroid Build Coastguard Worker fd = xsocket(AF_INET, SOCK_DGRAM, 0);
852*cf5a6c84SAndroid Build Coastguard Worker if (ioctl(fd, SIOCGIFTXQLEN, &req)) perror("");
853*cf5a6c84SAndroid Build Coastguard Worker else link->txqueuelen = req.ifr_ifru.ifru_ivalue;
854*cf5a6c84SAndroid Build Coastguard Worker xclose(fd);
855*cf5a6c84SAndroid Build Coastguard Worker }
856*cf5a6c84SAndroid Build Coastguard Worker
857*cf5a6c84SAndroid Build Coastguard Worker if (TT.is_addr && addrinfo.label && fnmatch(addrinfo.label, link->iface, 0))
858*cf5a6c84SAndroid Build Coastguard Worker return 0;
859*cf5a6c84SAndroid Build Coastguard Worker
860*cf5a6c84SAndroid Build Coastguard Worker
861*cf5a6c84SAndroid Build Coastguard Worker if (!(flags = get_flag_string(iface_flags, link->flags, 1)))
862*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid data.");
863*cf5a6c84SAndroid Build Coastguard Worker if (!TT.singleline) line_feed="\n ";
864*cf5a6c84SAndroid Build Coastguard Worker if (link->parent != -1) {
865*cf5a6c84SAndroid Build Coastguard Worker char iface[IF_NAMESIZE];
866*cf5a6c84SAndroid Build Coastguard Worker
867*cf5a6c84SAndroid Build Coastguard Worker if (!if_indextoname(link->parent, iface)) perror_exit(NULL);
868*cf5a6c84SAndroid Build Coastguard Worker sprintf(toybuf,"%s@%s", link->iface, iface);
869*cf5a6c84SAndroid Build Coastguard Worker }
870*cf5a6c84SAndroid Build Coastguard Worker if (link->flags & IFF_POINTOPOINT) peer = "peer";
871*cf5a6c84SAndroid Build Coastguard Worker if (TT.is_addr && TT.singleline && TT.addressfamily)
872*cf5a6c84SAndroid Build Coastguard Worker xprintf("%d: %s", link->iface_idx,
873*cf5a6c84SAndroid Build Coastguard Worker ((link->parent == -1) ? link->iface : toybuf));
874*cf5a6c84SAndroid Build Coastguard Worker else xprintf("%d: %s: <%s> mtu %d qdisc %s state %s qlen %d",
875*cf5a6c84SAndroid Build Coastguard Worker link->iface_idx, ((link->parent == -1) ? link->iface : toybuf), flags,
876*cf5a6c84SAndroid Build Coastguard Worker link->mtu, link->qdiscpline, link->state, link->txqueuelen);
877*cf5a6c84SAndroid Build Coastguard Worker
878*cf5a6c84SAndroid Build Coastguard Worker if (!TT.addressfamily || TT.addressfamily == AF_PACKET)
879*cf5a6c84SAndroid Build Coastguard Worker xprintf("%slink/%s %s %s %s",
880*cf5a6c84SAndroid Build Coastguard Worker line_feed, link->type, link->laddr, peer ,link->bcast);
881*cf5a6c84SAndroid Build Coastguard Worker
882*cf5a6c84SAndroid Build Coastguard Worker xputc('\n');
883*cf5a6c84SAndroid Build Coastguard Worker
884*cf5a6c84SAndroid Build Coastguard Worker //user can specify stats flag two times
885*cf5a6c84SAndroid Build Coastguard Worker //one for stats and other for erros e.g. -s and -s -s
886*cf5a6c84SAndroid Build Coastguard Worker print_stats(&link->rt_stat);
887*cf5a6c84SAndroid Build Coastguard Worker free(flags);
888*cf5a6c84SAndroid Build Coastguard Worker
889*cf5a6c84SAndroid Build Coastguard Worker return 0;
890*cf5a6c84SAndroid Build Coastguard Worker }
891*cf5a6c84SAndroid Build Coastguard Worker
fill_address(void * p,char * ip)892*cf5a6c84SAndroid Build Coastguard Worker static void fill_address(void *p, char *ip)
893*cf5a6c84SAndroid Build Coastguard Worker {
894*cf5a6c84SAndroid Build Coastguard Worker unsigned char *ptr = (unsigned char*)p;
895*cf5a6c84SAndroid Build Coastguard Worker snprintf(ip, 64, " %02x:%02x:%02x:%02x:%02x:%02x",
896*cf5a6c84SAndroid Build Coastguard Worker ptr[0], ptr[1], ptr[2], ptr[3], ptr[4], ptr[5]);
897*cf5a6c84SAndroid Build Coastguard Worker }
898*cf5a6c84SAndroid Build Coastguard Worker
get_link_info(struct nlmsghdr * h,struct linkdata * link,char ** argv)899*cf5a6c84SAndroid Build Coastguard Worker static int get_link_info(struct nlmsghdr* h,struct linkdata* link,char **argv)
900*cf5a6c84SAndroid Build Coastguard Worker {
901*cf5a6c84SAndroid Build Coastguard Worker struct ifinfomsg *iface = NLMSG_DATA(h);
902*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *attr = IFLA_RTA(iface);
903*cf5a6c84SAndroid Build Coastguard Worker int len = h->nlmsg_len - NLMSG_LENGTH(sizeof(*iface));
904*cf5a6c84SAndroid Build Coastguard Worker struct arglist hwtypes[]={{"generic",0},{"ether",ARPHRD_ETHER},
905*cf5a6c84SAndroid Build Coastguard Worker {"loopback", ARPHRD_LOOPBACK},{"sit",ARPHRD_SIT},
906*cf5a6c84SAndroid Build Coastguard Worker #ifdef ARPHRD_INFINIBAND
907*cf5a6c84SAndroid Build Coastguard Worker {"infiniband",ARPHRD_INFINIBAND},
908*cf5a6c84SAndroid Build Coastguard Worker #endif
909*cf5a6c84SAndroid Build Coastguard Worker #ifdef ARPHRD_IEEE802_TR
910*cf5a6c84SAndroid Build Coastguard Worker {"ieee802",ARPHRD_IEEE802}, {"tr",ARPHRD_IEEE802_TR},
911*cf5a6c84SAndroid Build Coastguard Worker #else
912*cf5a6c84SAndroid Build Coastguard Worker {"tr",ARPHRD_IEEE802},
913*cf5a6c84SAndroid Build Coastguard Worker #endif
914*cf5a6c84SAndroid Build Coastguard Worker #ifdef ARPHRD_IEEE80211
915*cf5a6c84SAndroid Build Coastguard Worker {"ieee802.11",ARPHRD_IEEE80211},
916*cf5a6c84SAndroid Build Coastguard Worker #endif
917*cf5a6c84SAndroid Build Coastguard Worker #ifdef ARPHRD_IEEE1394
918*cf5a6c84SAndroid Build Coastguard Worker {"ieee1394",ARPHRD_IEEE1394},
919*cf5a6c84SAndroid Build Coastguard Worker #endif
920*cf5a6c84SAndroid Build Coastguard Worker {"irda",ARPHRD_IRDA},{"slip",ARPHRD_SLIP},{"cslip",ARPHRD_CSLIP},
921*cf5a6c84SAndroid Build Coastguard Worker {"slip6",ARPHRD_SLIP6}, {"cslip6",ARPHRD_CSLIP6}, {"ppp",ARPHRD_PPP},
922*cf5a6c84SAndroid Build Coastguard Worker {"ipip",ARPHRD_TUNNEL}, {"tunnel6",ARPHRD_TUNNEL6},
923*cf5a6c84SAndroid Build Coastguard Worker {"gre",ARPHRD_IPGRE},
924*cf5a6c84SAndroid Build Coastguard Worker #ifdef ARPHRD_VOID
925*cf5a6c84SAndroid Build Coastguard Worker {"void",ARPHRD_VOID},
926*cf5a6c84SAndroid Build Coastguard Worker #endif
927*cf5a6c84SAndroid Build Coastguard Worker {NULL,-1}};
928*cf5a6c84SAndroid Build Coastguard Worker char *lname = get_flag_string(hwtypes, iface->ifi_type, 0);
929*cf5a6c84SAndroid Build Coastguard Worker
930*cf5a6c84SAndroid Build Coastguard Worker link->next = link->prev = 0;
931*cf5a6c84SAndroid Build Coastguard Worker link->iface_type = iface->ifi_type;
932*cf5a6c84SAndroid Build Coastguard Worker if (!lname) error_exit("Invalid link.");
933*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(link->type, lname, IFNAMSIZ);
934*cf5a6c84SAndroid Build Coastguard Worker free(lname);
935*cf5a6c84SAndroid Build Coastguard Worker link->iface_idx = iface->ifi_index;
936*cf5a6c84SAndroid Build Coastguard Worker link->flags = iface->ifi_flags;
937*cf5a6c84SAndroid Build Coastguard Worker if (*argv && !strcasecmp("up",*argv) && !(link->flags & IFF_UP)) return 1;
938*cf5a6c84SAndroid Build Coastguard Worker link->parent = -1;
939*cf5a6c84SAndroid Build Coastguard Worker for (; RTA_OK(attr, len); attr = RTA_NEXT(attr, len)) {
940*cf5a6c84SAndroid Build Coastguard Worker switch(attr->rta_type) {
941*cf5a6c84SAndroid Build Coastguard Worker case IFLA_IFNAME:
942*cf5a6c84SAndroid Build Coastguard Worker snprintf(link->iface, IFNAMSIZ, "%s",(char *) RTA_DATA(attr));
943*cf5a6c84SAndroid Build Coastguard Worker break;
944*cf5a6c84SAndroid Build Coastguard Worker case IFLA_ADDRESS:
945*cf5a6c84SAndroid Build Coastguard Worker if ( iface->ifi_type== ARPHRD_TUNNEL ||
946*cf5a6c84SAndroid Build Coastguard Worker iface->ifi_type == ARPHRD_SIT ||
947*cf5a6c84SAndroid Build Coastguard Worker iface->ifi_type == ARPHRD_IPGRE)
948*cf5a6c84SAndroid Build Coastguard Worker inet_ntop(AF_INET, RTA_DATA(attr), link->laddr, 64);
949*cf5a6c84SAndroid Build Coastguard Worker else fill_address(RTA_DATA(attr), link->laddr);
950*cf5a6c84SAndroid Build Coastguard Worker break;
951*cf5a6c84SAndroid Build Coastguard Worker case IFLA_BROADCAST:
952*cf5a6c84SAndroid Build Coastguard Worker if (iface->ifi_type== ARPHRD_TUNNEL ||
953*cf5a6c84SAndroid Build Coastguard Worker iface->ifi_type == ARPHRD_SIT ||
954*cf5a6c84SAndroid Build Coastguard Worker iface->ifi_type == ARPHRD_IPGRE)
955*cf5a6c84SAndroid Build Coastguard Worker inet_ntop(AF_INET, RTA_DATA(attr), link->bcast, 64);
956*cf5a6c84SAndroid Build Coastguard Worker else fill_address(RTA_DATA(attr), link->bcast);
957*cf5a6c84SAndroid Build Coastguard Worker break;
958*cf5a6c84SAndroid Build Coastguard Worker case IFLA_MTU:
959*cf5a6c84SAndroid Build Coastguard Worker link->mtu = *((int*)(RTA_DATA(attr)));
960*cf5a6c84SAndroid Build Coastguard Worker break;
961*cf5a6c84SAndroid Build Coastguard Worker case IFLA_QDISC:
962*cf5a6c84SAndroid Build Coastguard Worker snprintf(link->qdiscpline, IFNAMSIZ, "%s", (char *) RTA_DATA(attr));
963*cf5a6c84SAndroid Build Coastguard Worker break;
964*cf5a6c84SAndroid Build Coastguard Worker case IFLA_STATS :
965*cf5a6c84SAndroid Build Coastguard Worker link->rt_stat = *((struct rtnl_link_stats*) RTA_DATA(attr));
966*cf5a6c84SAndroid Build Coastguard Worker break;
967*cf5a6c84SAndroid Build Coastguard Worker case IFLA_LINK:
968*cf5a6c84SAndroid Build Coastguard Worker link->parent = *((int*)(RTA_DATA(attr)));
969*cf5a6c84SAndroid Build Coastguard Worker break;
970*cf5a6c84SAndroid Build Coastguard Worker case IFLA_TXQLEN:
971*cf5a6c84SAndroid Build Coastguard Worker link->txqueuelen = *((int*)(RTA_DATA(attr)));
972*cf5a6c84SAndroid Build Coastguard Worker break;
973*cf5a6c84SAndroid Build Coastguard Worker case IFLA_OPERSTATE:
974*cf5a6c84SAndroid Build Coastguard Worker {
975*cf5a6c84SAndroid Build Coastguard Worker struct arglist flags[]={{"UNKNOWN", 0}, {"NOTPRESENT", 1},
976*cf5a6c84SAndroid Build Coastguard Worker {"DOWN", 2}, {"LOWERLAYERDOWN", 3}, {"TESTING", 4},
977*cf5a6c84SAndroid Build Coastguard Worker {"DORMANT", 5}, {"UP", 6}, {NULL, -1}};
978*cf5a6c84SAndroid Build Coastguard Worker if (!(lname = get_flag_string(flags, *((int*)(RTA_DATA(attr))), 0)))
979*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid state.");
980*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(link->state, lname,IFNAMSIZ);
981*cf5a6c84SAndroid Build Coastguard Worker free(lname);
982*cf5a6c84SAndroid Build Coastguard Worker }
983*cf5a6c84SAndroid Build Coastguard Worker break;
984*cf5a6c84SAndroid Build Coastguard Worker default: break;
985*cf5a6c84SAndroid Build Coastguard Worker }
986*cf5a6c84SAndroid Build Coastguard Worker }
987*cf5a6c84SAndroid Build Coastguard Worker return 0;
988*cf5a6c84SAndroid Build Coastguard Worker }
989*cf5a6c84SAndroid Build Coastguard Worker
display_link_info(struct nlmsghdr * mhdr,char ** argv)990*cf5a6c84SAndroid Build Coastguard Worker static int display_link_info(struct nlmsghdr *mhdr, char **argv)
991*cf5a6c84SAndroid Build Coastguard Worker {
992*cf5a6c84SAndroid Build Coastguard Worker struct linkdata link;
993*cf5a6c84SAndroid Build Coastguard Worker
994*cf5a6c84SAndroid Build Coastguard Worker if (!get_link_info(mhdr, &link, argv)) {
995*cf5a6c84SAndroid Build Coastguard Worker if (TT.is_addr) {
996*cf5a6c84SAndroid Build Coastguard Worker struct linkdata *lnk = xzalloc(sizeof(struct linkdata));
997*cf5a6c84SAndroid Build Coastguard Worker memcpy(lnk, &link, sizeof(struct linkdata));
998*cf5a6c84SAndroid Build Coastguard Worker dlist_add_nomalloc((struct double_list **)&linfo,
999*cf5a6c84SAndroid Build Coastguard Worker (struct double_list *)lnk);
1000*cf5a6c84SAndroid Build Coastguard Worker }
1001*cf5a6c84SAndroid Build Coastguard Worker else print_link_output(&link);
1002*cf5a6c84SAndroid Build Coastguard Worker }
1003*cf5a6c84SAndroid Build Coastguard Worker return 0;
1004*cf5a6c84SAndroid Build Coastguard Worker }
1005*cf5a6c84SAndroid Build Coastguard Worker
link_show(char ** argv)1006*cf5a6c84SAndroid Build Coastguard Worker static int link_show(char **argv)
1007*cf5a6c84SAndroid Build Coastguard Worker {
1008*cf5a6c84SAndroid Build Coastguard Worker struct {
1009*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr mhdr;
1010*cf5a6c84SAndroid Build Coastguard Worker struct ifinfomsg info;
1011*cf5a6c84SAndroid Build Coastguard Worker } request;
1012*cf5a6c84SAndroid Build Coastguard Worker uint32_t index = 0;
1013*cf5a6c84SAndroid Build Coastguard Worker
1014*cf5a6c84SAndroid Build Coastguard Worker if (*argv && strcasecmp("up",*argv)) index = get_ifaceindex(*argv, 1);
1015*cf5a6c84SAndroid Build Coastguard Worker memset(&request, 0, sizeof(request));
1016*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_len = NLMSG_LENGTH(sizeof(struct ifinfomsg));
1017*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_flags = NLM_F_REQUEST|NLM_F_ACK;
1018*cf5a6c84SAndroid Build Coastguard Worker if (!index) request.mhdr.nlmsg_flags |= NLM_F_ROOT|NLM_F_MATCH;
1019*cf5a6c84SAndroid Build Coastguard Worker else request.info.ifi_change = 0xffffffff; // used in single operation
1020*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_type = RTM_GETLINK;
1021*cf5a6c84SAndroid Build Coastguard Worker request.info.ifi_index = index;
1022*cf5a6c84SAndroid Build Coastguard Worker request.info.ifi_family = AF_UNSPEC;
1023*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(0, 0, 0, (void*)&request, sizeof(request));
1024*cf5a6c84SAndroid Build Coastguard Worker return (filter_nlmesg(display_link_info, argv));
1025*cf5a6c84SAndroid Build Coastguard Worker }
1026*cf5a6c84SAndroid Build Coastguard Worker
iplink(char ** argv)1027*cf5a6c84SAndroid Build Coastguard Worker static int iplink(char **argv)
1028*cf5a6c84SAndroid Build Coastguard Worker {
1029*cf5a6c84SAndroid Build Coastguard Worker int idx;
1030*cf5a6c84SAndroid Build Coastguard Worker cmdobj ipcmd, cmdobjlist[] = {linkupdate, link_set, link_show};
1031*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist[] = {{"add", 0}, {"delete", 0},
1032*cf5a6c84SAndroid Build Coastguard Worker {"set", 1}, {"show", 2}, {"list", 2}, {"lst", 2}, {NULL,-1}};
1033*cf5a6c84SAndroid Build Coastguard Worker
1034*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) idx = 2;
1035*cf5a6c84SAndroid Build Coastguard Worker else if ((idx = substring_to_idx(*argv++, cmd_objectlist)) == -1)
1036*cf5a6c84SAndroid Build Coastguard Worker help_exit(0);
1037*cf5a6c84SAndroid Build Coastguard Worker ipcmd = cmdobjlist[idx];
1038*cf5a6c84SAndroid Build Coastguard Worker return ipcmd(argv);
1039*cf5a6c84SAndroid Build Coastguard Worker }
1040*cf5a6c84SAndroid Build Coastguard Worker
1041*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
1042*cf5a6c84SAndroid Build Coastguard Worker // Code for ip addr.
1043*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
1044*cf5a6c84SAndroid Build Coastguard Worker
print_addrinfo(struct nlmsghdr * h,int flag_l)1045*cf5a6c84SAndroid Build Coastguard Worker static int print_addrinfo(struct nlmsghdr *h, int flag_l)
1046*cf5a6c84SAndroid Build Coastguard Worker {
1047*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *rta, *rta_tb[IFA_MAX+1] = {0,};
1048*cf5a6c84SAndroid Build Coastguard Worker char *family = toybuf, *scope = toybuf+256, *label = toybuf+512,
1049*cf5a6c84SAndroid Build Coastguard Worker *brd = toybuf+768, *peer = toybuf+1024, *any = toybuf+1280,
1050*cf5a6c84SAndroid Build Coastguard Worker lbuf[INET6_ADDRSTRLEN] = {0,}, lbuf_ifa[INET6_ADDRSTRLEN] = {0,};
1051*cf5a6c84SAndroid Build Coastguard Worker struct ifaddrmsg *ifa = NLMSG_DATA(h);
1052*cf5a6c84SAndroid Build Coastguard Worker int len;
1053*cf5a6c84SAndroid Build Coastguard Worker
1054*cf5a6c84SAndroid Build Coastguard Worker if ((len = h->nlmsg_len - NLMSG_LENGTH(sizeof(*ifa))) < 0) {
1055*cf5a6c84SAndroid Build Coastguard Worker error_msg("wrong nlmsg len %d", len);
1056*cf5a6c84SAndroid Build Coastguard Worker return 0;
1057*cf5a6c84SAndroid Build Coastguard Worker }
1058*cf5a6c84SAndroid Build Coastguard Worker
1059*cf5a6c84SAndroid Build Coastguard Worker for (rta = IFA_RTA(ifa); RTA_OK(rta, len); rta=RTA_NEXT(rta, len))
1060*cf5a6c84SAndroid Build Coastguard Worker if (rta->rta_type <= IFA_MAX) rta_tb[rta->rta_type] = rta;
1061*cf5a6c84SAndroid Build Coastguard Worker
1062*cf5a6c84SAndroid Build Coastguard Worker if (!rta_tb[IFA_LOCAL]) rta_tb[IFA_LOCAL] = rta_tb[IFA_ADDRESS];
1063*cf5a6c84SAndroid Build Coastguard Worker if (!rta_tb[IFA_ADDRESS]) rta_tb[IFA_ADDRESS] = rta_tb[IFA_LOCAL];
1064*cf5a6c84SAndroid Build Coastguard Worker if ((addrinfo.scope ^ ifa->ifa_scope)&addrinfo.scopemask) return 0;
1065*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.ifindex && addrinfo.ifindex != ifa->ifa_index) return 0;
1066*cf5a6c84SAndroid Build Coastguard Worker
1067*cf5a6c84SAndroid Build Coastguard Worker if (flag_l && addrinfo.label && ifa->ifa_family == AF_INET6) return 0;
1068*cf5a6c84SAndroid Build Coastguard Worker if ((rta_tb[IFA_LABEL])) {
1069*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(label, RTA_DATA(rta_tb[IFA_LABEL]), 256);
1070*cf5a6c84SAndroid Build Coastguard Worker label[255] = '\0';
1071*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.label && fnmatch(addrinfo.label, label, 0))
1072*cf5a6c84SAndroid Build Coastguard Worker return 0;
1073*cf5a6c84SAndroid Build Coastguard Worker }
1074*cf5a6c84SAndroid Build Coastguard Worker
1075*cf5a6c84SAndroid Build Coastguard Worker if (TT.flush) {
1076*cf5a6c84SAndroid Build Coastguard Worker if (ifa->ifa_index == addrinfo.ifindex) {
1077*cf5a6c84SAndroid Build Coastguard Worker h->nlmsg_type = RTM_DELADDR;
1078*cf5a6c84SAndroid Build Coastguard Worker h->nlmsg_flags = NLM_F_REQUEST;
1079*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(RTM_DELADDR, 0, 0, h, h->nlmsg_len);
1080*cf5a6c84SAndroid Build Coastguard Worker return 0;
1081*cf5a6c84SAndroid Build Coastguard Worker }
1082*cf5a6c84SAndroid Build Coastguard Worker }
1083*cf5a6c84SAndroid Build Coastguard Worker
1084*cf5a6c84SAndroid Build Coastguard Worker if (h->nlmsg_type == RTM_DELADDR) printf("Deleted ");
1085*cf5a6c84SAndroid Build Coastguard Worker
1086*cf5a6c84SAndroid Build Coastguard Worker if (TT.singleline) {
1087*cf5a6c84SAndroid Build Coastguard Worker if (!if_indextoname(ifa->ifa_index, lbuf)) perror_exit(NULL);
1088*cf5a6c84SAndroid Build Coastguard Worker printf("%u: %s",ifa->ifa_index, lbuf);
1089*cf5a6c84SAndroid Build Coastguard Worker }
1090*cf5a6c84SAndroid Build Coastguard Worker
1091*cf5a6c84SAndroid Build Coastguard Worker sprintf(scope, " scope %s ", namefromRPDB(ifa->ifa_scope, RPDB_rtscopes));
1092*cf5a6c84SAndroid Build Coastguard Worker
1093*cf5a6c84SAndroid Build Coastguard Worker if (ifa->ifa_family == AF_INET) strcpy(family, " inet ");
1094*cf5a6c84SAndroid Build Coastguard Worker else if (ifa->ifa_family == AF_INET6) strcpy(family, " inet6 ");
1095*cf5a6c84SAndroid Build Coastguard Worker else sprintf(family, " family %d", ifa->ifa_family);
1096*cf5a6c84SAndroid Build Coastguard Worker
1097*cf5a6c84SAndroid Build Coastguard Worker if (rta_tb[IFA_LOCAL]) {
1098*cf5a6c84SAndroid Build Coastguard Worker if (!inet_ntop(ifa->ifa_family, RTA_DATA(rta_tb[IFA_LOCAL]),
1099*cf5a6c84SAndroid Build Coastguard Worker lbuf, sizeof(lbuf))) perror_exit("inet");
1100*cf5a6c84SAndroid Build Coastguard Worker
1101*cf5a6c84SAndroid Build Coastguard Worker sprintf(family+strlen(family), lbuf, strlen(lbuf));
1102*cf5a6c84SAndroid Build Coastguard Worker if (!rta_tb[IFA_ADDRESS] || !memcmp(RTA_DATA(rta_tb[IFA_ADDRESS]),
1103*cf5a6c84SAndroid Build Coastguard Worker RTA_DATA(rta_tb[IFA_LOCAL]), 4))
1104*cf5a6c84SAndroid Build Coastguard Worker sprintf(family+strlen(family), "/%d ", ifa->ifa_prefixlen);
1105*cf5a6c84SAndroid Build Coastguard Worker else {
1106*cf5a6c84SAndroid Build Coastguard Worker if (!inet_ntop(ifa->ifa_family, RTA_DATA(rta_tb[IFA_ADDRESS]),
1107*cf5a6c84SAndroid Build Coastguard Worker lbuf_ifa, sizeof(lbuf_ifa))) perror_exit("inet");
1108*cf5a6c84SAndroid Build Coastguard Worker sprintf(peer, " peer %s/%d ", lbuf_ifa, ifa->ifa_prefixlen);
1109*cf5a6c84SAndroid Build Coastguard Worker }
1110*cf5a6c84SAndroid Build Coastguard Worker }
1111*cf5a6c84SAndroid Build Coastguard Worker
1112*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.to && strcmp(addrinfo.addr, lbuf))
1113*cf5a6c84SAndroid Build Coastguard Worker return 0;
1114*cf5a6c84SAndroid Build Coastguard Worker
1115*cf5a6c84SAndroid Build Coastguard Worker if (rta_tb[IFA_BROADCAST]) {
1116*cf5a6c84SAndroid Build Coastguard Worker if (!inet_ntop(ifa->ifa_family, RTA_DATA(rta_tb[IFA_BROADCAST]),
1117*cf5a6c84SAndroid Build Coastguard Worker lbuf, sizeof(lbuf))) perror_exit("inet");
1118*cf5a6c84SAndroid Build Coastguard Worker sprintf(brd, " brd %s", lbuf);
1119*cf5a6c84SAndroid Build Coastguard Worker }else brd = "";
1120*cf5a6c84SAndroid Build Coastguard Worker
1121*cf5a6c84SAndroid Build Coastguard Worker if (rta_tb[IFA_ANYCAST]) {
1122*cf5a6c84SAndroid Build Coastguard Worker if (!inet_ntop(ifa->ifa_family, RTA_DATA(rta_tb[IFA_ANYCAST]),
1123*cf5a6c84SAndroid Build Coastguard Worker lbuf, sizeof(lbuf))) perror_exit("inet");
1124*cf5a6c84SAndroid Build Coastguard Worker sprintf(any, " any %s", lbuf);
1125*cf5a6c84SAndroid Build Coastguard Worker }
1126*cf5a6c84SAndroid Build Coastguard Worker
1127*cf5a6c84SAndroid Build Coastguard Worker if (ifa->ifa_family == AF_INET)
1128*cf5a6c84SAndroid Build Coastguard Worker printf("%s%s%s%s%s %c", family, brd, peer, scope, label,
1129*cf5a6c84SAndroid Build Coastguard Worker (TT.singleline? '\0' : '\n'));
1130*cf5a6c84SAndroid Build Coastguard Worker else printf("%s%s %c", family, scope, (TT.singleline? '\0' : '\n'));
1131*cf5a6c84SAndroid Build Coastguard Worker if (TT.singleline && (ifa->ifa_family == AF_INET)) xputc('\n');
1132*cf5a6c84SAndroid Build Coastguard Worker
1133*cf5a6c84SAndroid Build Coastguard Worker if (rta_tb[IFA_CACHEINFO]) {
1134*cf5a6c84SAndroid Build Coastguard Worker struct ifa_cacheinfo *ci = RTA_DATA(rta_tb[IFA_CACHEINFO]);
1135*cf5a6c84SAndroid Build Coastguard Worker
1136*cf5a6c84SAndroid Build Coastguard Worker printf("%c valid_lft ", (TT.singleline? '\\' : '\0'));
1137*cf5a6c84SAndroid Build Coastguard Worker if (ci->ifa_valid == 0xFFFFFFFFU) printf("forever");
1138*cf5a6c84SAndroid Build Coastguard Worker else printf("%usec", ci->ifa_valid);
1139*cf5a6c84SAndroid Build Coastguard Worker printf(" preferred_lft ");
1140*cf5a6c84SAndroid Build Coastguard Worker if (ci->ifa_prefered == 0xFFFFFFFFU) printf("forever");
1141*cf5a6c84SAndroid Build Coastguard Worker else printf("%dsec", ci->ifa_prefered);
1142*cf5a6c84SAndroid Build Coastguard Worker xputc('\n');
1143*cf5a6c84SAndroid Build Coastguard Worker }
1144*cf5a6c84SAndroid Build Coastguard Worker return 0;
1145*cf5a6c84SAndroid Build Coastguard Worker }
1146*cf5a6c84SAndroid Build Coastguard Worker
ipaddrupdate(char ** argv)1147*cf5a6c84SAndroid Build Coastguard Worker static int ipaddrupdate(char **argv)
1148*cf5a6c84SAndroid Build Coastguard Worker {
1149*cf5a6c84SAndroid Build Coastguard Worker int length, cmd = !memcmp("add", argv[-1], strlen(argv[-1]))
1150*cf5a6c84SAndroid Build Coastguard Worker ? RTM_NEWADDR: RTM_DELADDR;
1151*cf5a6c84SAndroid Build Coastguard Worker int idx = 0,length_brd = 0, length_peer = 0,length_any = 0,length_local = 0,
1152*cf5a6c84SAndroid Build Coastguard Worker scoped = 0;
1153*cf5a6c84SAndroid Build Coastguard Worker char *dev = NULL,*label = NULL, reply[8192];
1154*cf5a6c84SAndroid Build Coastguard Worker
1155*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr *addr_ptr = NULL;
1156*cf5a6c84SAndroid Build Coastguard Worker struct nlmsgerr *err = NULL;
1157*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist[] = {{"dev",0}, {"peer", 1},
1158*cf5a6c84SAndroid Build Coastguard Worker {"remote", 2}, {"broadcast", 3}, {"brd", 4}, {"label", 5},
1159*cf5a6c84SAndroid Build Coastguard Worker {"anycast", 6},{"scope", 7}, {"local", 8}, {NULL, -1}};
1160*cf5a6c84SAndroid Build Coastguard Worker struct {
1161*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr nlm;
1162*cf5a6c84SAndroid Build Coastguard Worker struct ifaddrmsg ifadd;
1163*cf5a6c84SAndroid Build Coastguard Worker char buf[256];
1164*cf5a6c84SAndroid Build Coastguard Worker } req;
1165*cf5a6c84SAndroid Build Coastguard Worker typedef struct {
1166*cf5a6c84SAndroid Build Coastguard Worker int family, bytelen, bitlen;
1167*cf5a6c84SAndroid Build Coastguard Worker __u32 data[8];
1168*cf5a6c84SAndroid Build Coastguard Worker } option_data;
1169*cf5a6c84SAndroid Build Coastguard Worker option_data local;
1170*cf5a6c84SAndroid Build Coastguard Worker
1171*cf5a6c84SAndroid Build Coastguard Worker memset(&req, 0, sizeof(req));
1172*cf5a6c84SAndroid Build Coastguard Worker req.nlm.nlmsg_len = NLMSG_LENGTH(sizeof(struct ifaddrmsg));
1173*cf5a6c84SAndroid Build Coastguard Worker req.nlm.nlmsg_flags = NLM_F_REQUEST|NLM_F_ACK;
1174*cf5a6c84SAndroid Build Coastguard Worker req.nlm.nlmsg_type = cmd;
1175*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_family = TT.addressfamily;
1176*cf5a6c84SAndroid Build Coastguard Worker
1177*cf5a6c84SAndroid Build Coastguard Worker while (*argv) {
1178*cf5a6c84SAndroid Build Coastguard Worker idx = substring_to_idx(*argv, cmd_objectlist);
1179*cf5a6c84SAndroid Build Coastguard Worker if (idx >= 0)
1180*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv)
1181*cf5a6c84SAndroid Build Coastguard Worker error_exit("Incomplete Command line");
1182*cf5a6c84SAndroid Build Coastguard Worker switch(idx) {
1183*cf5a6c84SAndroid Build Coastguard Worker case 0:
1184*cf5a6c84SAndroid Build Coastguard Worker dev = *argv;
1185*cf5a6c84SAndroid Build Coastguard Worker break;
1186*cf5a6c84SAndroid Build Coastguard Worker case 1:
1187*cf5a6c84SAndroid Build Coastguard Worker case 2:
1188*cf5a6c84SAndroid Build Coastguard Worker {
1189*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[4] = {0,}, netmask = 0;
1190*cf5a6c84SAndroid Build Coastguard Worker uint8_t len = 0;
1191*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(addr, &netmask, &len, *argv,
1192*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_family);
1193*cf5a6c84SAndroid Build Coastguard Worker if (len)
1194*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_family = ((len == 4) ? AF_INET : AF_INET6);
1195*cf5a6c84SAndroid Build Coastguard Worker length_peer = len;
1196*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.nlm, sizeof(req),
1197*cf5a6c84SAndroid Build Coastguard Worker IFA_ADDRESS, addr, len);
1198*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_prefixlen = netmask;
1199*cf5a6c84SAndroid Build Coastguard Worker }
1200*cf5a6c84SAndroid Build Coastguard Worker break;
1201*cf5a6c84SAndroid Build Coastguard Worker case 3:
1202*cf5a6c84SAndroid Build Coastguard Worker case 4:
1203*cf5a6c84SAndroid Build Coastguard Worker if (*argv[0] == '+') {
1204*cf5a6c84SAndroid Build Coastguard Worker length_brd = -1;
1205*cf5a6c84SAndroid Build Coastguard Worker } else if (*argv[0] == '-') {
1206*cf5a6c84SAndroid Build Coastguard Worker length_brd = -2;
1207*cf5a6c84SAndroid Build Coastguard Worker } else {
1208*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[4] = {0,};
1209*cf5a6c84SAndroid Build Coastguard Worker uint8_t af = AF_UNSPEC;
1210*cf5a6c84SAndroid Build Coastguard Worker
1211*cf5a6c84SAndroid Build Coastguard Worker if (get_prefix(addr, &af, *argv, req.ifadd.ifa_family))
1212*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid prefix");
1213*cf5a6c84SAndroid Build Coastguard Worker
1214*cf5a6c84SAndroid Build Coastguard Worker length_brd = ((af == AF_INET6) ? 16 : 4);
1215*cf5a6c84SAndroid Build Coastguard Worker if (req.ifadd.ifa_family == AF_UNSPEC)
1216*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_family = af;
1217*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.nlm, sizeof(req),
1218*cf5a6c84SAndroid Build Coastguard Worker IFA_BROADCAST, &addr, length_brd);
1219*cf5a6c84SAndroid Build Coastguard Worker }
1220*cf5a6c84SAndroid Build Coastguard Worker break;
1221*cf5a6c84SAndroid Build Coastguard Worker case 5:
1222*cf5a6c84SAndroid Build Coastguard Worker label = *argv;
1223*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.nlm, sizeof(req),
1224*cf5a6c84SAndroid Build Coastguard Worker IFA_LABEL, label, strlen(label) + 1);
1225*cf5a6c84SAndroid Build Coastguard Worker break;
1226*cf5a6c84SAndroid Build Coastguard Worker case 6:
1227*cf5a6c84SAndroid Build Coastguard Worker {
1228*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[4] = {0,};
1229*cf5a6c84SAndroid Build Coastguard Worker uint8_t af = AF_UNSPEC;
1230*cf5a6c84SAndroid Build Coastguard Worker
1231*cf5a6c84SAndroid Build Coastguard Worker if (get_prefix(addr, &af, *argv, req.ifadd.ifa_family))
1232*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid prefix");
1233*cf5a6c84SAndroid Build Coastguard Worker length_any = ((af == AF_INET6) ? 16 : 4);
1234*cf5a6c84SAndroid Build Coastguard Worker if (req.ifadd.ifa_family == AF_UNSPEC)
1235*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_family = af;
1236*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.nlm, sizeof(req),
1237*cf5a6c84SAndroid Build Coastguard Worker IFA_ANYCAST, &addr, length_any);
1238*cf5a6c84SAndroid Build Coastguard Worker }
1239*cf5a6c84SAndroid Build Coastguard Worker break;
1240*cf5a6c84SAndroid Build Coastguard Worker case 7:
1241*cf5a6c84SAndroid Build Coastguard Worker {
1242*cf5a6c84SAndroid Build Coastguard Worker int scope = idxfromRPDB(*argv, RPDB_rtscopes);
1243*cf5a6c84SAndroid Build Coastguard Worker if (scope < 0) error_exit("wrong scope '%s'", *argv);
1244*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_scope = scope;
1245*cf5a6c84SAndroid Build Coastguard Worker scoped = 1;
1246*cf5a6c84SAndroid Build Coastguard Worker }
1247*cf5a6c84SAndroid Build Coastguard Worker break;
1248*cf5a6c84SAndroid Build Coastguard Worker default:
1249*cf5a6c84SAndroid Build Coastguard Worker {
1250*cf5a6c84SAndroid Build Coastguard Worker //local is by default
1251*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[8] = {0,}, netmask = 0;
1252*cf5a6c84SAndroid Build Coastguard Worker uint8_t len = 0;
1253*cf5a6c84SAndroid Build Coastguard Worker
1254*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(addr, &netmask, &len, *argv,
1255*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_family);
1256*cf5a6c84SAndroid Build Coastguard Worker if (len)
1257*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_family = ((len == 4) ? AF_INET : AF_INET6);
1258*cf5a6c84SAndroid Build Coastguard Worker length_local = len;
1259*cf5a6c84SAndroid Build Coastguard Worker local.bitlen = netmask;
1260*cf5a6c84SAndroid Build Coastguard Worker local.bytelen = len;
1261*cf5a6c84SAndroid Build Coastguard Worker memcpy(local.data, addr, sizeof(local.data));
1262*cf5a6c84SAndroid Build Coastguard Worker local.family = req.ifadd.ifa_family;
1263*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.nlm, sizeof(req),
1264*cf5a6c84SAndroid Build Coastguard Worker IFA_LOCAL, &local.data, local.bytelen);
1265*cf5a6c84SAndroid Build Coastguard Worker }
1266*cf5a6c84SAndroid Build Coastguard Worker break;
1267*cf5a6c84SAndroid Build Coastguard Worker }
1268*cf5a6c84SAndroid Build Coastguard Worker argv++;
1269*cf5a6c84SAndroid Build Coastguard Worker }
1270*cf5a6c84SAndroid Build Coastguard Worker if (!dev) error_exit("need \"dev \" argument");
1271*cf5a6c84SAndroid Build Coastguard Worker if (label && strncmp(dev, label, strlen(dev)) != 0)
1272*cf5a6c84SAndroid Build Coastguard Worker error_exit("\"dev\" (%s) must match \"label\" (%s)", dev, label);
1273*cf5a6c84SAndroid Build Coastguard Worker
1274*cf5a6c84SAndroid Build Coastguard Worker if (length_peer == 0 && length_local && cmd != RTM_DELADDR){
1275*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.nlm, sizeof(req),
1276*cf5a6c84SAndroid Build Coastguard Worker IFA_ADDRESS, &local.data, local.bytelen);
1277*cf5a6c84SAndroid Build Coastguard Worker }
1278*cf5a6c84SAndroid Build Coastguard Worker
1279*cf5a6c84SAndroid Build Coastguard Worker if (length_brd < 0 && cmd != RTM_DELADDR){
1280*cf5a6c84SAndroid Build Coastguard Worker int i;
1281*cf5a6c84SAndroid Build Coastguard Worker
1282*cf5a6c84SAndroid Build Coastguard Worker if (req.ifadd.ifa_family != AF_INET)
1283*cf5a6c84SAndroid Build Coastguard Worker error_exit("broadcast can be set only for IPv4 addresses");
1284*cf5a6c84SAndroid Build Coastguard Worker
1285*cf5a6c84SAndroid Build Coastguard Worker if (local.bitlen <= 30) {
1286*cf5a6c84SAndroid Build Coastguard Worker for (i = 31; i >= local.bitlen; i--) {
1287*cf5a6c84SAndroid Build Coastguard Worker if (length_brd == -1)
1288*cf5a6c84SAndroid Build Coastguard Worker local.data[0] |= htonl(1<<(31-i));
1289*cf5a6c84SAndroid Build Coastguard Worker else
1290*cf5a6c84SAndroid Build Coastguard Worker local.data[0] &= ~htonl(1<<(31-i));
1291*cf5a6c84SAndroid Build Coastguard Worker }
1292*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.nlm, sizeof(req),
1293*cf5a6c84SAndroid Build Coastguard Worker IFA_BROADCAST, &local.data, local.bytelen);
1294*cf5a6c84SAndroid Build Coastguard Worker length_brd = local.bytelen;
1295*cf5a6c84SAndroid Build Coastguard Worker }
1296*cf5a6c84SAndroid Build Coastguard Worker }
1297*cf5a6c84SAndroid Build Coastguard Worker if (req.ifadd.ifa_prefixlen == 0)
1298*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_prefixlen = local.bitlen;
1299*cf5a6c84SAndroid Build Coastguard Worker if (!scoped && (cmd != RTM_DELADDR) && (local.family == AF_INET)
1300*cf5a6c84SAndroid Build Coastguard Worker && (local.bytelen >= 1 && *(uint8_t*)&local.data == 127))
1301*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_scope = RT_SCOPE_HOST;
1302*cf5a6c84SAndroid Build Coastguard Worker req.ifadd.ifa_index = get_ifaceindex(dev, 1);
1303*cf5a6c84SAndroid Build Coastguard Worker
1304*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(RTM_NEWADDR, 0, AF_UNSPEC, (void *)&req, req.nlm.nlmsg_len);
1305*cf5a6c84SAndroid Build Coastguard Worker length = recv(TT.sockfd, reply, sizeof(reply), 0);
1306*cf5a6c84SAndroid Build Coastguard Worker addr_ptr = (struct nlmsghdr *) reply;
1307*cf5a6c84SAndroid Build Coastguard Worker for (; NLMSG_OK(addr_ptr, length); addr_ptr = NLMSG_NEXT(addr_ptr, length)) {
1308*cf5a6c84SAndroid Build Coastguard Worker if (addr_ptr->nlmsg_type == NLMSG_DONE)
1309*cf5a6c84SAndroid Build Coastguard Worker return 1;
1310*cf5a6c84SAndroid Build Coastguard Worker if (addr_ptr->nlmsg_type == NLMSG_ERROR)
1311*cf5a6c84SAndroid Build Coastguard Worker err = (struct nlmsgerr*) NLMSG_DATA(addr_ptr);
1312*cf5a6c84SAndroid Build Coastguard Worker if (err && err->error) {
1313*cf5a6c84SAndroid Build Coastguard Worker errno = -err->error;
1314*cf5a6c84SAndroid Build Coastguard Worker perror_exit("RTNETLINK answers:");
1315*cf5a6c84SAndroid Build Coastguard Worker }
1316*cf5a6c84SAndroid Build Coastguard Worker }
1317*cf5a6c84SAndroid Build Coastguard Worker return 0;
1318*cf5a6c84SAndroid Build Coastguard Worker }
1319*cf5a6c84SAndroid Build Coastguard Worker
ipaddr_listflush(char ** argv)1320*cf5a6c84SAndroid Build Coastguard Worker static int ipaddr_listflush(char **argv)
1321*cf5a6c84SAndroid Build Coastguard Worker {
1322*cf5a6c84SAndroid Build Coastguard Worker int idx; uint32_t netmask = 0, found = 0;
1323*cf5a6c84SAndroid Build Coastguard Worker char *tmp = NULL, *name = NULL;
1324*cf5a6c84SAndroid Build Coastguard Worker struct double_list *dlist;
1325*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist[] = {{"to", 0}, {"scope", 1}, {"up", 2},
1326*cf5a6c84SAndroid Build Coastguard Worker {"label", 3}, {"dev", 4}, {NULL, -1}};
1327*cf5a6c84SAndroid Build Coastguard Worker
1328*cf5a6c84SAndroid Build Coastguard Worker TT.flush = *argv[-1] == 'f' ? 1 : 0;
1329*cf5a6c84SAndroid Build Coastguard Worker memset(&addrinfo, 0, sizeof(addrinfo));
1330*cf5a6c84SAndroid Build Coastguard Worker
1331*cf5a6c84SAndroid Build Coastguard Worker if (TT.flush) {
1332*cf5a6c84SAndroid Build Coastguard Worker if (!*argv)
1333*cf5a6c84SAndroid Build Coastguard Worker error_exit("Incomplete command for \"flush\"");
1334*cf5a6c84SAndroid Build Coastguard Worker if (TT.addressfamily == AF_PACKET)
1335*cf5a6c84SAndroid Build Coastguard Worker error_exit("Can't flush link Addresses");
1336*cf5a6c84SAndroid Build Coastguard Worker }
1337*cf5a6c84SAndroid Build Coastguard Worker addrinfo.scope = -1;
1338*cf5a6c84SAndroid Build Coastguard Worker while (*argv) {
1339*cf5a6c84SAndroid Build Coastguard Worker switch (idx = substring_to_idx(*argv, cmd_objectlist)) {
1340*cf5a6c84SAndroid Build Coastguard Worker case 0:
1341*cf5a6c84SAndroid Build Coastguard Worker {// ADDR_TO
1342*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Incomplete Command line");
1343*cf5a6c84SAndroid Build Coastguard Worker else if (!strcmp(*argv, "0")) return 0;
1344*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[4] = {0,};
1345*cf5a6c84SAndroid Build Coastguard Worker uint8_t len = 0;
1346*cf5a6c84SAndroid Build Coastguard Worker
1347*cf5a6c84SAndroid Build Coastguard Worker addrinfo.to = 1;
1348*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(addr, &netmask, &len, *argv, TT.addressfamily);
1349*cf5a6c84SAndroid Build Coastguard Worker if (len)
1350*cf5a6c84SAndroid Build Coastguard Worker TT.addressfamily = ((len == 4) ? AF_INET : AF_INET6);
1351*cf5a6c84SAndroid Build Coastguard Worker addrinfo.addr = strtok(*argv, "/");
1352*cf5a6c84SAndroid Build Coastguard Worker }
1353*cf5a6c84SAndroid Build Coastguard Worker break;
1354*cf5a6c84SAndroid Build Coastguard Worker case 1: // ADDR_SCOPE
1355*cf5a6c84SAndroid Build Coastguard Worker {
1356*cf5a6c84SAndroid Build Coastguard Worker int scope = 0;
1357*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Incomplete Command line");
1358*cf5a6c84SAndroid Build Coastguard Worker name = *argv;
1359*cf5a6c84SAndroid Build Coastguard Worker
1360*cf5a6c84SAndroid Build Coastguard Worker addrinfo.scopemask = -1;
1361*cf5a6c84SAndroid Build Coastguard Worker if (isdigit(**argv)) {
1362*cf5a6c84SAndroid Build Coastguard Worker int idx = atolx(*argv);
1363*cf5a6c84SAndroid Build Coastguard Worker
1364*cf5a6c84SAndroid Build Coastguard Worker name = xstrdup(namefromRPDB(idx, RPDB_rtscopes));
1365*cf5a6c84SAndroid Build Coastguard Worker }
1366*cf5a6c84SAndroid Build Coastguard Worker if ((scope = idxfromRPDB(name, RPDB_rtscopes)) < 0) {
1367*cf5a6c84SAndroid Build Coastguard Worker if (strcmp(name, "all"))
1368*cf5a6c84SAndroid Build Coastguard Worker error_exit("wrong scope '%s'", name);
1369*cf5a6c84SAndroid Build Coastguard Worker scope = RT_SCOPE_NOWHERE;
1370*cf5a6c84SAndroid Build Coastguard Worker addrinfo.scopemask = 0;
1371*cf5a6c84SAndroid Build Coastguard Worker }
1372*cf5a6c84SAndroid Build Coastguard Worker
1373*cf5a6c84SAndroid Build Coastguard Worker if (isdigit(**argv))
1374*cf5a6c84SAndroid Build Coastguard Worker free(name);
1375*cf5a6c84SAndroid Build Coastguard Worker addrinfo.scope = scope;
1376*cf5a6c84SAndroid Build Coastguard Worker }
1377*cf5a6c84SAndroid Build Coastguard Worker break;
1378*cf5a6c84SAndroid Build Coastguard Worker case 2: // ADDR_UP
1379*cf5a6c84SAndroid Build Coastguard Worker addrinfo.up = 1;
1380*cf5a6c84SAndroid Build Coastguard Worker break;
1381*cf5a6c84SAndroid Build Coastguard Worker case 3: // ADDR_LABEL
1382*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Incomplete Command line");
1383*cf5a6c84SAndroid Build Coastguard Worker addrinfo.label = *argv;
1384*cf5a6c84SAndroid Build Coastguard Worker break;
1385*cf5a6c84SAndroid Build Coastguard Worker case 4: // ADDR_DEV
1386*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Incomplete Command line");
1387*cf5a6c84SAndroid Build Coastguard Worker
1388*cf5a6c84SAndroid Build Coastguard Worker default:
1389*cf5a6c84SAndroid Build Coastguard Worker if (TT.filter_dev)
1390*cf5a6c84SAndroid Build Coastguard Worker error_exit("Either \"dev\" is duplicate or %s is garbage",
1391*cf5a6c84SAndroid Build Coastguard Worker *argv);
1392*cf5a6c84SAndroid Build Coastguard Worker TT.filter_dev = *argv;
1393*cf5a6c84SAndroid Build Coastguard Worker break;
1394*cf5a6c84SAndroid Build Coastguard Worker }
1395*cf5a6c84SAndroid Build Coastguard Worker argv++;
1396*cf5a6c84SAndroid Build Coastguard Worker }
1397*cf5a6c84SAndroid Build Coastguard Worker
1398*cf5a6c84SAndroid Build Coastguard Worker link_show(&tmp);
1399*cf5a6c84SAndroid Build Coastguard Worker while ( linfo && (dlist = dlist_pop(&linfo))){
1400*cf5a6c84SAndroid Build Coastguard Worker struct linkdata *tmp = (struct linkdata*) dlist;
1401*cf5a6c84SAndroid Build Coastguard Worker char *temp = &tmp->iface[0];
1402*cf5a6c84SAndroid Build Coastguard Worker
1403*cf5a6c84SAndroid Build Coastguard Worker if (TT.filter_dev && strcmp(TT.filter_dev, temp))
1404*cf5a6c84SAndroid Build Coastguard Worker continue;
1405*cf5a6c84SAndroid Build Coastguard Worker found = 1;
1406*cf5a6c84SAndroid Build Coastguard Worker if (TT.flush && addrinfo.label) ipaddr_print( tmp, 0);
1407*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.up && !(tmp->flags & IFF_UP)){
1408*cf5a6c84SAndroid Build Coastguard Worker ipaddr_print(tmp, 0);
1409*cf5a6c84SAndroid Build Coastguard Worker continue;
1410*cf5a6c84SAndroid Build Coastguard Worker }
1411*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.label){
1412*cf5a6c84SAndroid Build Coastguard Worker if ( fnmatch(addrinfo.label, temp, 0)) {
1413*cf5a6c84SAndroid Build Coastguard Worker ipaddr_print(tmp, 1);
1414*cf5a6c84SAndroid Build Coastguard Worker continue;
1415*cf5a6c84SAndroid Build Coastguard Worker }
1416*cf5a6c84SAndroid Build Coastguard Worker }
1417*cf5a6c84SAndroid Build Coastguard Worker if (!TT.addressfamily && ! TT.flush ) print_link_output(tmp);
1418*cf5a6c84SAndroid Build Coastguard Worker
1419*cf5a6c84SAndroid Build Coastguard Worker ipaddr_print(tmp, 0);
1420*cf5a6c84SAndroid Build Coastguard Worker free(tmp);
1421*cf5a6c84SAndroid Build Coastguard Worker }
1422*cf5a6c84SAndroid Build Coastguard Worker if (TT.filter_dev && !found)
1423*cf5a6c84SAndroid Build Coastguard Worker error_exit("Device \"%s\" doesn't exist. \n", TT.filter_dev);
1424*cf5a6c84SAndroid Build Coastguard Worker return 0;
1425*cf5a6c84SAndroid Build Coastguard Worker }
1426*cf5a6c84SAndroid Build Coastguard Worker
ipaddr_print(struct linkdata * link,int flag_l)1427*cf5a6c84SAndroid Build Coastguard Worker static int ipaddr_print( struct linkdata *link, int flag_l)
1428*cf5a6c84SAndroid Build Coastguard Worker {
1429*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr *addr_ptr;
1430*cf5a6c84SAndroid Build Coastguard Worker int ip_match = 0;
1431*cf5a6c84SAndroid Build Coastguard Worker
1432*cf5a6c84SAndroid Build Coastguard Worker addrinfo.ifindex = link->iface_idx;
1433*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(RTM_GETADDR, NLM_F_ROOT|NLM_F_MATCH|NLM_F_REQUEST,
1434*cf5a6c84SAndroid Build Coastguard Worker AF_UNSPEC, NULL, 0);
1435*cf5a6c84SAndroid Build Coastguard Worker if (TT.addressfamily == AF_PACKET) print_link_output(link);
1436*cf5a6c84SAndroid Build Coastguard Worker
1437*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.label){
1438*cf5a6c84SAndroid Build Coastguard Worker char *col = strchr(addrinfo.label, ':');
1439*cf5a6c84SAndroid Build Coastguard Worker if (!col && (fnmatch(addrinfo.label, &link->iface[0], 0)))
1440*cf5a6c84SAndroid Build Coastguard Worker return 0;
1441*cf5a6c84SAndroid Build Coastguard Worker }
1442*cf5a6c84SAndroid Build Coastguard Worker
1443*cf5a6c84SAndroid Build Coastguard Worker while (1){
1444*cf5a6c84SAndroid Build Coastguard Worker int len = recv(TT.sockfd, TT.gbuf, sizeof(TT.gbuf), 0);
1445*cf5a6c84SAndroid Build Coastguard Worker addr_ptr = (struct nlmsghdr *)TT.gbuf;
1446*cf5a6c84SAndroid Build Coastguard Worker struct ifaddrmsg *addressInfo = NLMSG_DATA(addr_ptr);
1447*cf5a6c84SAndroid Build Coastguard Worker char lbuf[INET6_ADDRSTRLEN];
1448*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *rta, *rta_tb[IFA_MAX+1] = {0,};
1449*cf5a6c84SAndroid Build Coastguard Worker
1450*cf5a6c84SAndroid Build Coastguard Worker int len1 = addr_ptr->nlmsg_len - NLMSG_LENGTH(sizeof(*addressInfo));
1451*cf5a6c84SAndroid Build Coastguard Worker if (len1 > 0) {
1452*cf5a6c84SAndroid Build Coastguard Worker for (; NLMSG_OK(addr_ptr, len); addr_ptr = NLMSG_NEXT(addr_ptr, len)) {
1453*cf5a6c84SAndroid Build Coastguard Worker addressInfo = NLMSG_DATA(addr_ptr);
1454*cf5a6c84SAndroid Build Coastguard Worker if (TT.addressfamily && TT.addressfamily != addressInfo->ifa_family)
1455*cf5a6c84SAndroid Build Coastguard Worker continue;
1456*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.ifindex && addrinfo.ifindex != addressInfo->ifa_index)
1457*cf5a6c84SAndroid Build Coastguard Worker continue;
1458*cf5a6c84SAndroid Build Coastguard Worker
1459*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.to) {
1460*cf5a6c84SAndroid Build Coastguard Worker memset(rta_tb, 0, sizeof(rta_tb));
1461*cf5a6c84SAndroid Build Coastguard Worker int rt_len = IFA_PAYLOAD(addr_ptr);
1462*cf5a6c84SAndroid Build Coastguard Worker for (rta = IFA_RTA(addressInfo); RTA_OK(rta, rt_len); rta=RTA_NEXT(rta, rt_len)) {
1463*cf5a6c84SAndroid Build Coastguard Worker if (rta->rta_type <= IFA_MAX) rta_tb[rta->rta_type] = rta;
1464*cf5a6c84SAndroid Build Coastguard Worker }
1465*cf5a6c84SAndroid Build Coastguard Worker if (!rta_tb[IFA_LOCAL]) rta_tb[IFA_LOCAL] = rta_tb[IFA_ADDRESS];
1466*cf5a6c84SAndroid Build Coastguard Worker if (rta_tb[IFA_LOCAL]) {
1467*cf5a6c84SAndroid Build Coastguard Worker if (!inet_ntop(TT.addressfamily, RTA_DATA(rta_tb[IFA_LOCAL]),
1468*cf5a6c84SAndroid Build Coastguard Worker lbuf, sizeof(lbuf))) perror_exit("inet");
1469*cf5a6c84SAndroid Build Coastguard Worker if (strcmp(addrinfo.addr, lbuf))
1470*cf5a6c84SAndroid Build Coastguard Worker continue;
1471*cf5a6c84SAndroid Build Coastguard Worker ip_match=1;
1472*cf5a6c84SAndroid Build Coastguard Worker }
1473*cf5a6c84SAndroid Build Coastguard Worker if (!ip_match)
1474*cf5a6c84SAndroid Build Coastguard Worker continue;
1475*cf5a6c84SAndroid Build Coastguard Worker }
1476*cf5a6c84SAndroid Build Coastguard Worker
1477*cf5a6c84SAndroid Build Coastguard Worker if (!TT.flush){
1478*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.scope != -1 && TT.addressfamily && TT.addressfamily ==
1479*cf5a6c84SAndroid Build Coastguard Worker addressInfo->ifa_family &&
1480*cf5a6c84SAndroid Build Coastguard Worker (addrinfo.ifindex == addressInfo->ifa_index)) {
1481*cf5a6c84SAndroid Build Coastguard Worker if ((addrinfo.scope ^ addressInfo->ifa_scope) & addrinfo.scopemask)
1482*cf5a6c84SAndroid Build Coastguard Worker continue;
1483*cf5a6c84SAndroid Build Coastguard Worker else if (addrinfo.up && (link->flags & IFF_UP))
1484*cf5a6c84SAndroid Build Coastguard Worker print_link_output(link);
1485*cf5a6c84SAndroid Build Coastguard Worker else if (!addrinfo.up) print_link_output(link);
1486*cf5a6c84SAndroid Build Coastguard Worker }
1487*cf5a6c84SAndroid Build Coastguard Worker if (TT.addressfamily &&
1488*cf5a6c84SAndroid Build Coastguard Worker (addrinfo.ifindex == addressInfo->ifa_index) &&
1489*cf5a6c84SAndroid Build Coastguard Worker (addrinfo.scope == -1)){
1490*cf5a6c84SAndroid Build Coastguard Worker if (addrinfo.up && (link->flags & IFF_UP))
1491*cf5a6c84SAndroid Build Coastguard Worker print_link_output(link);
1492*cf5a6c84SAndroid Build Coastguard Worker else if (!addrinfo.up) print_link_output(link);
1493*cf5a6c84SAndroid Build Coastguard Worker }
1494*cf5a6c84SAndroid Build Coastguard Worker }
1495*cf5a6c84SAndroid Build Coastguard Worker
1496*cf5a6c84SAndroid Build Coastguard Worker for (; NLMSG_OK(addr_ptr, len); addr_ptr = NLMSG_NEXT(addr_ptr, len)) {
1497*cf5a6c84SAndroid Build Coastguard Worker if (addr_ptr->nlmsg_type == RTM_NEWADDR)
1498*cf5a6c84SAndroid Build Coastguard Worker print_addrinfo(addr_ptr, flag_l);
1499*cf5a6c84SAndroid Build Coastguard Worker if ((addr_ptr->nlmsg_type == NLMSG_DONE) ||
1500*cf5a6c84SAndroid Build Coastguard Worker (addr_ptr->nlmsg_type == NLMSG_ERROR) ||
1501*cf5a6c84SAndroid Build Coastguard Worker (TT.flush && addrinfo.to))
1502*cf5a6c84SAndroid Build Coastguard Worker goto ret_stop;
1503*cf5a6c84SAndroid Build Coastguard Worker }
1504*cf5a6c84SAndroid Build Coastguard Worker if ((addr_ptr->nlmsg_type == NLMSG_DONE) ||
1505*cf5a6c84SAndroid Build Coastguard Worker (addr_ptr->nlmsg_type == NLMSG_ERROR))
1506*cf5a6c84SAndroid Build Coastguard Worker break;
1507*cf5a6c84SAndroid Build Coastguard Worker }
1508*cf5a6c84SAndroid Build Coastguard Worker }
1509*cf5a6c84SAndroid Build Coastguard Worker else
1510*cf5a6c84SAndroid Build Coastguard Worker return 0;
1511*cf5a6c84SAndroid Build Coastguard Worker }
1512*cf5a6c84SAndroid Build Coastguard Worker
1513*cf5a6c84SAndroid Build Coastguard Worker ret_stop:
1514*cf5a6c84SAndroid Build Coastguard Worker return 0;
1515*cf5a6c84SAndroid Build Coastguard Worker }
1516*cf5a6c84SAndroid Build Coastguard Worker
ipaddr(char ** argv)1517*cf5a6c84SAndroid Build Coastguard Worker static int ipaddr(char **argv)
1518*cf5a6c84SAndroid Build Coastguard Worker {
1519*cf5a6c84SAndroid Build Coastguard Worker int idx;
1520*cf5a6c84SAndroid Build Coastguard Worker cmdobj ipcmd, cmdobjlist[] = {ipaddrupdate, ipaddr_listflush};
1521*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist[] = { {"add", 0}, {"delete", 0},
1522*cf5a6c84SAndroid Build Coastguard Worker {"list", 1},{"show", 1},{"lst", 1}, {"flush", 1}, {NULL,-1}};
1523*cf5a6c84SAndroid Build Coastguard Worker
1524*cf5a6c84SAndroid Build Coastguard Worker TT.is_addr++;
1525*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) idx = 1;
1526*cf5a6c84SAndroid Build Coastguard Worker else if ((idx = substring_to_idx(*argv++, cmd_objectlist)) == -1)
1527*cf5a6c84SAndroid Build Coastguard Worker help_exit(0);
1528*cf5a6c84SAndroid Build Coastguard Worker
1529*cf5a6c84SAndroid Build Coastguard Worker ipcmd = cmdobjlist[idx];
1530*cf5a6c84SAndroid Build Coastguard Worker return ipcmd(argv);
1531*cf5a6c84SAndroid Build Coastguard Worker }
1532*cf5a6c84SAndroid Build Coastguard Worker
1533*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
1534*cf5a6c84SAndroid Build Coastguard Worker // code for ip route
1535*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
1536*cf5a6c84SAndroid Build Coastguard Worker struct I_data {
1537*cf5a6c84SAndroid Build Coastguard Worker unsigned char family;
1538*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[8] , netmask ;
1539*cf5a6c84SAndroid Build Coastguard Worker uint8_t len ;
1540*cf5a6c84SAndroid Build Coastguard Worker };
1541*cf5a6c84SAndroid Build Coastguard Worker
1542*cf5a6c84SAndroid Build Coastguard Worker struct {
1543*cf5a6c84SAndroid Build Coastguard Worker int tb,idev,odev,proto;
1544*cf5a6c84SAndroid Build Coastguard Worker struct I_data rvia, rdst, mdst, rsrc, msrc;
1545*cf5a6c84SAndroid Build Coastguard Worker } gfilter;
1546*cf5a6c84SAndroid Build Coastguard Worker
show_iproute_help(void)1547*cf5a6c84SAndroid Build Coastguard Worker static void show_iproute_help(void)
1548*cf5a6c84SAndroid Build Coastguard Worker {
1549*cf5a6c84SAndroid Build Coastguard Worker error_exit("\n\n" \
1550*cf5a6c84SAndroid Build Coastguard Worker "iproute { list | flush } SELECTOR\n" \
1551*cf5a6c84SAndroid Build Coastguard Worker "iproute get ADDRESS [from ADDRESS iif STRING]\n" \
1552*cf5a6c84SAndroid Build Coastguard Worker " [oif STRING]\n" \
1553*cf5a6c84SAndroid Build Coastguard Worker "iproute { add | del | change | append | replace | test } ROUTE\n" \
1554*cf5a6c84SAndroid Build Coastguard Worker " SELECTOR := [root PREFIX] [match PREFIX] [proto RTPROTO]\n" \
1555*cf5a6c84SAndroid Build Coastguard Worker " ROUTE := [TYPE] PREFIX [proto RTPROTO] [metric METRIC]");
1556*cf5a6c84SAndroid Build Coastguard Worker }
1557*cf5a6c84SAndroid Build Coastguard Worker
print_rta_metrics(char * out,const struct rtattr * mxattr)1558*cf5a6c84SAndroid Build Coastguard Worker static void print_rta_metrics(char* out, const struct rtattr *mxattr)
1559*cf5a6c84SAndroid Build Coastguard Worker {
1560*cf5a6c84SAndroid Build Coastguard Worker int32_t tvar = RTA_PAYLOAD(mxattr);
1561*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *rta, *mxrta[RTAX_MAX+1] = {0,};
1562*cf5a6c84SAndroid Build Coastguard Worker unsigned int mxlock = 0;
1563*cf5a6c84SAndroid Build Coastguard Worker int i;
1564*cf5a6c84SAndroid Build Coastguard Worker
1565*cf5a6c84SAndroid Build Coastguard Worker for (rta = RTA_DATA(mxattr); RTA_OK(rta, tvar); rta=RTA_NEXT(rta, tvar))
1566*cf5a6c84SAndroid Build Coastguard Worker if (rta->rta_type <= RTA_MAX) mxrta[rta->rta_type] = rta;
1567*cf5a6c84SAndroid Build Coastguard Worker
1568*cf5a6c84SAndroid Build Coastguard Worker if (mxrta[RTAX_LOCK])
1569*cf5a6c84SAndroid Build Coastguard Worker mxlock = *(u_int32_t *)RTA_DATA(mxrta[RTAX_LOCK]);
1570*cf5a6c84SAndroid Build Coastguard Worker
1571*cf5a6c84SAndroid Build Coastguard Worker for (i = 2; i <= RTAX_MAX; i++) {
1572*cf5a6c84SAndroid Build Coastguard Worker uint32_t val = 0;
1573*cf5a6c84SAndroid Build Coastguard Worker
1574*cf5a6c84SAndroid Build Coastguard Worker if (mxrta[i] == NULL && !(mxlock & (1 << i)))
1575*cf5a6c84SAndroid Build Coastguard Worker continue;
1576*cf5a6c84SAndroid Build Coastguard Worker
1577*cf5a6c84SAndroid Build Coastguard Worker if (mxrta[i] != NULL && i != RTAX_CC_ALGO)
1578*cf5a6c84SAndroid Build Coastguard Worker val = *(u_int32_t *)RTA_DATA(mxrta[i]);
1579*cf5a6c84SAndroid Build Coastguard Worker
1580*cf5a6c84SAndroid Build Coastguard Worker if (i == RTAX_HOPLIMIT && (int)val == -1)
1581*cf5a6c84SAndroid Build Coastguard Worker continue;
1582*cf5a6c84SAndroid Build Coastguard Worker
1583*cf5a6c84SAndroid Build Coastguard Worker if (i < sizeof(mx_names)/sizeof(char *) && mx_names[i])
1584*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s%s ", out, mx_names[i]);
1585*cf5a6c84SAndroid Build Coastguard Worker else
1586*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%smetric %d ", out, i);
1587*cf5a6c84SAndroid Build Coastguard Worker
1588*cf5a6c84SAndroid Build Coastguard Worker if (mxlock & (1<<i))
1589*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%slock ", out);
1590*cf5a6c84SAndroid Build Coastguard Worker
1591*cf5a6c84SAndroid Build Coastguard Worker switch (i) {
1592*cf5a6c84SAndroid Build Coastguard Worker case RTAX_RTT:
1593*cf5a6c84SAndroid Build Coastguard Worker case RTAX_RTTVAR:
1594*cf5a6c84SAndroid Build Coastguard Worker case RTAX_RTO_MIN:
1595*cf5a6c84SAndroid Build Coastguard Worker if (i == RTAX_RTT)
1596*cf5a6c84SAndroid Build Coastguard Worker val /= 8;
1597*cf5a6c84SAndroid Build Coastguard Worker else if (i == RTAX_RTTVAR)
1598*cf5a6c84SAndroid Build Coastguard Worker val /= 4;
1599*cf5a6c84SAndroid Build Coastguard Worker
1600*cf5a6c84SAndroid Build Coastguard Worker if (val >= 1000)
1601*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s%gs ", out, val / 1e3);
1602*cf5a6c84SAndroid Build Coastguard Worker else
1603*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s%ums ", out, val);
1604*cf5a6c84SAndroid Build Coastguard Worker break;
1605*cf5a6c84SAndroid Build Coastguard Worker
1606*cf5a6c84SAndroid Build Coastguard Worker case RTAX_CC_ALGO:
1607*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%scongestion %s ", out, (const char*)RTA_DATA(mxrta[i]));
1608*cf5a6c84SAndroid Build Coastguard Worker break;
1609*cf5a6c84SAndroid Build Coastguard Worker
1610*cf5a6c84SAndroid Build Coastguard Worker default:
1611*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s%u ", out, val);
1612*cf5a6c84SAndroid Build Coastguard Worker break;
1613*cf5a6c84SAndroid Build Coastguard Worker }
1614*cf5a6c84SAndroid Build Coastguard Worker }
1615*cf5a6c84SAndroid Build Coastguard Worker }
1616*cf5a6c84SAndroid Build Coastguard Worker
display_route_info(struct nlmsghdr * mhdr,char ** argv)1617*cf5a6c84SAndroid Build Coastguard Worker static int display_route_info(struct nlmsghdr *mhdr, char **argv)
1618*cf5a6c84SAndroid Build Coastguard Worker {
1619*cf5a6c84SAndroid Build Coastguard Worker char *inetval = NULL, out[1024] = {0};
1620*cf5a6c84SAndroid Build Coastguard Worker struct rtmsg *msg = NLMSG_DATA(mhdr);
1621*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *rta, *attr[RTA_MAX+1] = {0,};
1622*cf5a6c84SAndroid Build Coastguard Worker int32_t tvar, msglen = mhdr->nlmsg_len - NLMSG_LENGTH(sizeof(struct rtmsg));
1623*cf5a6c84SAndroid Build Coastguard Worker int hlen = ((msg->rtm_family == AF_INET) ? 32
1624*cf5a6c84SAndroid Build Coastguard Worker : ((msg->rtm_family == AF_INET6) ? 128 : -1));
1625*cf5a6c84SAndroid Build Coastguard Worker
1626*cf5a6c84SAndroid Build Coastguard Worker if (mhdr->nlmsg_type != RTM_NEWROUTE) return 0;
1627*cf5a6c84SAndroid Build Coastguard Worker if (msglen < 0) return 1;
1628*cf5a6c84SAndroid Build Coastguard Worker
1629*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_family == AF_INET6) {
1630*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.tb) {
1631*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.tb < 0) {
1632*cf5a6c84SAndroid Build Coastguard Worker if (!(msg->rtm_flags & RTM_F_CLONED)) return 0;
1633*cf5a6c84SAndroid Build Coastguard Worker } else {
1634*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_flags & RTM_F_CLONED) return 0;
1635*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.tb == RT_TABLE_LOCAL && msg->rtm_type != RTN_LOCAL)
1636*cf5a6c84SAndroid Build Coastguard Worker return 0;
1637*cf5a6c84SAndroid Build Coastguard Worker else if (gfilter.tb == RT_TABLE_MAIN && msg->rtm_type == RTN_LOCAL)
1638*cf5a6c84SAndroid Build Coastguard Worker return 0;
1639*cf5a6c84SAndroid Build Coastguard Worker }
1640*cf5a6c84SAndroid Build Coastguard Worker }
1641*cf5a6c84SAndroid Build Coastguard Worker }
1642*cf5a6c84SAndroid Build Coastguard Worker else if (gfilter.tb > 0 && gfilter.tb != msg->rtm_table) return 0;
1643*cf5a6c84SAndroid Build Coastguard Worker
1644*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.proto && (msg->rtm_protocol != gfilter.proto)) return 0;
1645*cf5a6c84SAndroid Build Coastguard Worker
1646*cf5a6c84SAndroid Build Coastguard Worker
1647*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.rdst.family && (msg->rtm_family != gfilter.rdst.family ||
1648*cf5a6c84SAndroid Build Coastguard Worker gfilter.rdst.netmask > msg->rtm_dst_len)) return 0;
1649*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.mdst.family && (msg->rtm_family != gfilter.mdst.family
1650*cf5a6c84SAndroid Build Coastguard Worker || (gfilter.mdst.netmask < msg->rtm_dst_len))) return 0;
1651*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.rsrc.family && (msg->rtm_family != gfilter.rsrc.family
1652*cf5a6c84SAndroid Build Coastguard Worker || gfilter.rsrc.netmask > msg->rtm_src_len)) return 0;
1653*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.msrc.family && (msg->rtm_family != gfilter.msrc.family
1654*cf5a6c84SAndroid Build Coastguard Worker || (gfilter.msrc.netmask < msg->rtm_src_len))) return 0;
1655*cf5a6c84SAndroid Build Coastguard Worker tvar = msglen;
1656*cf5a6c84SAndroid Build Coastguard Worker
1657*cf5a6c84SAndroid Build Coastguard Worker for (rta = RTM_RTA(msg); RTA_OK(rta, tvar); rta=RTA_NEXT(rta, tvar))
1658*cf5a6c84SAndroid Build Coastguard Worker if (rta->rta_type <= RTA_MAX) attr[rta->rta_type] = rta;
1659*cf5a6c84SAndroid Build Coastguard Worker
1660*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_type != RTN_UNICAST)
1661*cf5a6c84SAndroid Build Coastguard Worker sprintf(out,"%s%s ", out,rtmtype_idx2str(msg->rtm_type));
1662*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_DST]) {
1663*cf5a6c84SAndroid Build Coastguard Worker inetval = (char *)inet_ntop(msg->rtm_family, RTA_DATA(attr[RTA_DST]),
1664*cf5a6c84SAndroid Build Coastguard Worker toybuf, sizeof(toybuf));
1665*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.rdst.family &&
1666*cf5a6c84SAndroid Build Coastguard Worker memcmp(RTA_DATA(attr[RTA_DST]), &gfilter.rdst.addr, gfilter.rdst.len))
1667*cf5a6c84SAndroid Build Coastguard Worker return 0;
1668*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.mdst.family &&
1669*cf5a6c84SAndroid Build Coastguard Worker memcmp(RTA_DATA(attr[RTA_DST]), &gfilter.mdst.addr, gfilter.mdst.len))
1670*cf5a6c84SAndroid Build Coastguard Worker return 0;
1671*cf5a6c84SAndroid Build Coastguard Worker sprintf(out,"%s%s",out,inetval);
1672*cf5a6c84SAndroid Build Coastguard Worker }
1673*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_dst_len) sprintf(out,"%s/%d ", out,msg->rtm_dst_len);
1674*cf5a6c84SAndroid Build Coastguard Worker else sprintf(out,"%s%s",out,"default ");
1675*cf5a6c84SAndroid Build Coastguard Worker
1676*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_SRC]) {
1677*cf5a6c84SAndroid Build Coastguard Worker inetval = (char *)inet_ntop(msg->rtm_family, RTA_DATA(attr[RTA_SRC]),
1678*cf5a6c84SAndroid Build Coastguard Worker toybuf, sizeof(toybuf));
1679*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.rsrc.family &&
1680*cf5a6c84SAndroid Build Coastguard Worker memcmp(RTA_DATA(attr[RTA_SRC]), &gfilter.rsrc.addr, gfilter.rsrc.len))
1681*cf5a6c84SAndroid Build Coastguard Worker return 0;
1682*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.msrc.family &&
1683*cf5a6c84SAndroid Build Coastguard Worker memcmp(RTA_DATA(attr[RTA_SRC]), &gfilter.msrc.addr, gfilter.msrc.len))
1684*cf5a6c84SAndroid Build Coastguard Worker return 0;
1685*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s from %s", out, inetval);
1686*cf5a6c84SAndroid Build Coastguard Worker }
1687*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_src_len) sprintf(out, "%s/%d ", out, msg->rtm_src_len);
1688*cf5a6c84SAndroid Build Coastguard Worker
1689*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_GATEWAY]) {
1690*cf5a6c84SAndroid Build Coastguard Worker inetval = (char *)inet_ntop(msg->rtm_family, RTA_DATA(attr[RTA_GATEWAY]),
1691*cf5a6c84SAndroid Build Coastguard Worker toybuf, sizeof(toybuf));
1692*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s via %s ", out, inetval);
1693*cf5a6c84SAndroid Build Coastguard Worker }
1694*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.rvia.family) {
1695*cf5a6c84SAndroid Build Coastguard Worker char tmp[256];
1696*cf5a6c84SAndroid Build Coastguard Worker
1697*cf5a6c84SAndroid Build Coastguard Worker if (!attr[RTA_GATEWAY]) return 0;
1698*cf5a6c84SAndroid Build Coastguard Worker if (strcmp((char *)inet_ntop(msg->rtm_family, gfilter.rvia.addr,
1699*cf5a6c84SAndroid Build Coastguard Worker tmp, sizeof(tmp)), inetval)) return 0;
1700*cf5a6c84SAndroid Build Coastguard Worker }
1701*cf5a6c84SAndroid Build Coastguard Worker
1702*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.odev != 0) if (!attr[RTA_OIF]) return 0;
1703*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_OIF]) {
1704*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.odev !=0 && gfilter.odev != *(int*)RTA_DATA(attr[RTA_OIF]))
1705*cf5a6c84SAndroid Build Coastguard Worker return 0;
1706*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s dev %s ", out,
1707*cf5a6c84SAndroid Build Coastguard Worker if_indextoname(*(int*)RTA_DATA(attr[RTA_OIF]), toybuf));
1708*cf5a6c84SAndroid Build Coastguard Worker }
1709*cf5a6c84SAndroid Build Coastguard Worker
1710*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_PREFSRC] && hlen) {
1711*cf5a6c84SAndroid Build Coastguard Worker inetval = (char *)inet_ntop(msg->rtm_family, RTA_DATA(attr[RTA_PREFSRC]),
1712*cf5a6c84SAndroid Build Coastguard Worker toybuf, sizeof(toybuf));
1713*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s src %s ", out, inetval);
1714*cf5a6c84SAndroid Build Coastguard Worker }
1715*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_PRIORITY])
1716*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s metric %d ", out, *(uint32_t*)RTA_DATA(attr[RTA_PRIORITY]));
1717*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_family == AF_INET6) {
1718*cf5a6c84SAndroid Build Coastguard Worker struct rta_cacheinfo *ci = NULL;
1719*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_CACHEINFO]) ci = RTA_DATA(attr[RTA_CACHEINFO]);
1720*cf5a6c84SAndroid Build Coastguard Worker if ((msg->rtm_flags & RTM_F_CLONED) || (ci && ci->rta_expires)) {
1721*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_flags & RTM_F_CLONED) sprintf(out, "%s%s cache ",
1722*cf5a6c84SAndroid Build Coastguard Worker out, (!TT.singleline ? "\n" : " "));
1723*cf5a6c84SAndroid Build Coastguard Worker if (ci && ci->rta_expires) {
1724*cf5a6c84SAndroid Build Coastguard Worker int hz = 0;
1725*cf5a6c84SAndroid Build Coastguard Worker FILE *fp = xfopen("/proc/net/psched","r");
1726*cf5a6c84SAndroid Build Coastguard Worker
1727*cf5a6c84SAndroid Build Coastguard Worker if (fp) {
1728*cf5a6c84SAndroid Build Coastguard Worker unsigned int nom, denom;
1729*cf5a6c84SAndroid Build Coastguard Worker
1730*cf5a6c84SAndroid Build Coastguard Worker if (fscanf(fp, "%*08x%*08x%08x%08x", &nom, &denom) == 2)
1731*cf5a6c84SAndroid Build Coastguard Worker if (nom == 1000000)
1732*cf5a6c84SAndroid Build Coastguard Worker hz = denom;
1733*cf5a6c84SAndroid Build Coastguard Worker fclose(fp);
1734*cf5a6c84SAndroid Build Coastguard Worker }
1735*cf5a6c84SAndroid Build Coastguard Worker if (!hz) hz = sysconf(_SC_CLK_TCK);
1736*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s expires %dsec", out, ci->rta_expires /hz);
1737*cf5a6c84SAndroid Build Coastguard Worker }
1738*cf5a6c84SAndroid Build Coastguard Worker if (ci && ci->rta_error) sprintf(out, "%s error %d", out, ci->rta_error);
1739*cf5a6c84SAndroid Build Coastguard Worker }
1740*cf5a6c84SAndroid Build Coastguard Worker else if (ci && ci->rta_error)
1741*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s error %d", out, ci->rta_error);
1742*cf5a6c84SAndroid Build Coastguard Worker }
1743*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_IIF] && !gfilter.idev)
1744*cf5a6c84SAndroid Build Coastguard Worker sprintf(out, "%s iif %s", out,
1745*cf5a6c84SAndroid Build Coastguard Worker if_indextoname(*(int*)RTA_DATA(attr[RTA_IIF]), toybuf));
1746*cf5a6c84SAndroid Build Coastguard Worker
1747*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_METRICS])
1748*cf5a6c84SAndroid Build Coastguard Worker print_rta_metrics(out, attr[RTA_METRICS]);
1749*cf5a6c84SAndroid Build Coastguard Worker
1750*cf5a6c84SAndroid Build Coastguard Worker if (TT.flush || (TT.connected && !TT.from_ok))
1751*cf5a6c84SAndroid Build Coastguard Worker memcpy(toybuf, (void*)mhdr,mhdr->nlmsg_len);
1752*cf5a6c84SAndroid Build Coastguard Worker
1753*cf5a6c84SAndroid Build Coastguard Worker if (TT.flush) {
1754*cf5a6c84SAndroid Build Coastguard Worker int sockfd = 0;
1755*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr* mhdr = (struct nlmsghdr*)toybuf;
1756*cf5a6c84SAndroid Build Coastguard Worker struct rtmsg *msg = NLMSG_DATA(mhdr);
1757*cf5a6c84SAndroid Build Coastguard Worker int tvar, msglen = mhdr->nlmsg_len - NLMSG_LENGTH(sizeof(struct rtmsg));
1758*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *rta, *attr[RTA_MAX+1] = {0,};
1759*cf5a6c84SAndroid Build Coastguard Worker
1760*cf5a6c84SAndroid Build Coastguard Worker tvar = msglen;
1761*cf5a6c84SAndroid Build Coastguard Worker for (rta = RTM_RTA(msg); RTA_OK(rta, tvar); rta=RTA_NEXT(rta, tvar))
1762*cf5a6c84SAndroid Build Coastguard Worker if (rta->rta_type <= RTA_MAX) attr[rta->rta_type] = rta;
1763*cf5a6c84SAndroid Build Coastguard Worker
1764*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_family == AF_INET6
1765*cf5a6c84SAndroid Build Coastguard Worker && !msg->rtm_dst_len
1766*cf5a6c84SAndroid Build Coastguard Worker && msg->rtm_type == RTN_UNREACHABLE
1767*cf5a6c84SAndroid Build Coastguard Worker && attr[RTA_PRIORITY]
1768*cf5a6c84SAndroid Build Coastguard Worker && *(int*)RTA_DATA(attr[RTA_PRIORITY]) == -1)
1769*cf5a6c84SAndroid Build Coastguard Worker return 0;
1770*cf5a6c84SAndroid Build Coastguard Worker
1771*cf5a6c84SAndroid Build Coastguard Worker mhdr->nlmsg_flags = NLM_F_REQUEST | NLM_F_ACK;
1772*cf5a6c84SAndroid Build Coastguard Worker mhdr->nlmsg_type = RTM_DELROUTE;
1773*cf5a6c84SAndroid Build Coastguard Worker mhdr->nlmsg_pid = 0;
1774*cf5a6c84SAndroid Build Coastguard Worker sockfd = xsocket(AF_NETLINK, SOCK_RAW, NETLINK_ROUTE);
1775*cf5a6c84SAndroid Build Coastguard Worker if (send(sockfd , (void*)mhdr, mhdr->nlmsg_len, 0) < 0)
1776*cf5a6c84SAndroid Build Coastguard Worker perror_exit("Unable to send data on socket.");
1777*cf5a6c84SAndroid Build Coastguard Worker
1778*cf5a6c84SAndroid Build Coastguard Worker while (1) {
1779*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr *mhdr;
1780*cf5a6c84SAndroid Build Coastguard Worker int msglen = recv(sockfd, toybuf, sizeof(toybuf), 0);
1781*cf5a6c84SAndroid Build Coastguard Worker
1782*cf5a6c84SAndroid Build Coastguard Worker if ((msglen < 0) && (errno == EINTR || errno == EAGAIN)) continue;
1783*cf5a6c84SAndroid Build Coastguard Worker else if (msglen < 0) {
1784*cf5a6c84SAndroid Build Coastguard Worker error_msg("netlink receive error %s", strerror(errno));
1785*cf5a6c84SAndroid Build Coastguard Worker xclose(sockfd);
1786*cf5a6c84SAndroid Build Coastguard Worker return 1;
1787*cf5a6c84SAndroid Build Coastguard Worker } else if (!msglen) {
1788*cf5a6c84SAndroid Build Coastguard Worker error_msg("EOF on netlink");
1789*cf5a6c84SAndroid Build Coastguard Worker xclose(sockfd);
1790*cf5a6c84SAndroid Build Coastguard Worker return 1;
1791*cf5a6c84SAndroid Build Coastguard Worker }
1792*cf5a6c84SAndroid Build Coastguard Worker
1793*cf5a6c84SAndroid Build Coastguard Worker for (mhdr = (struct nlmsghdr*)toybuf; NLMSG_OK(mhdr, msglen);
1794*cf5a6c84SAndroid Build Coastguard Worker mhdr = NLMSG_NEXT(mhdr, msglen)) {
1795*cf5a6c84SAndroid Build Coastguard Worker switch (mhdr->nlmsg_type) {
1796*cf5a6c84SAndroid Build Coastguard Worker case NLMSG_DONE:
1797*cf5a6c84SAndroid Build Coastguard Worker xclose(sockfd);
1798*cf5a6c84SAndroid Build Coastguard Worker return 0;
1799*cf5a6c84SAndroid Build Coastguard Worker case NLMSG_ERROR:
1800*cf5a6c84SAndroid Build Coastguard Worker {
1801*cf5a6c84SAndroid Build Coastguard Worker struct nlmsgerr *merr = (struct nlmsgerr*)NLMSG_DATA(mhdr);
1802*cf5a6c84SAndroid Build Coastguard Worker
1803*cf5a6c84SAndroid Build Coastguard Worker if (merr->error == 0) { xclose(sockfd); return 0; }
1804*cf5a6c84SAndroid Build Coastguard Worker if (mhdr->nlmsg_len < NLMSG_LENGTH(sizeof(struct nlmsgerr)))
1805*cf5a6c84SAndroid Build Coastguard Worker error_msg("ERROR truncated");
1806*cf5a6c84SAndroid Build Coastguard Worker else {
1807*cf5a6c84SAndroid Build Coastguard Worker errno = -merr->error;
1808*cf5a6c84SAndroid Build Coastguard Worker perror_msg("RTNETLINK answers");
1809*cf5a6c84SAndroid Build Coastguard Worker }
1810*cf5a6c84SAndroid Build Coastguard Worker xclose(sockfd);
1811*cf5a6c84SAndroid Build Coastguard Worker return 1;
1812*cf5a6c84SAndroid Build Coastguard Worker }
1813*cf5a6c84SAndroid Build Coastguard Worker default:
1814*cf5a6c84SAndroid Build Coastguard Worker break;
1815*cf5a6c84SAndroid Build Coastguard Worker }
1816*cf5a6c84SAndroid Build Coastguard Worker } // End of for loop.
1817*cf5a6c84SAndroid Build Coastguard Worker } // End of while loop.
1818*cf5a6c84SAndroid Build Coastguard Worker
1819*cf5a6c84SAndroid Build Coastguard Worker xclose(sockfd);
1820*cf5a6c84SAndroid Build Coastguard Worker } else printf("%s\n",out);
1821*cf5a6c84SAndroid Build Coastguard Worker return 0;
1822*cf5a6c84SAndroid Build Coastguard Worker }
1823*cf5a6c84SAndroid Build Coastguard Worker
route_get(char ** argv)1824*cf5a6c84SAndroid Build Coastguard Worker static int route_get(char **argv)
1825*cf5a6c84SAndroid Build Coastguard Worker {
1826*cf5a6c84SAndroid Build Coastguard Worker int idx, flag;
1827*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist[] = {{"from", 0}, {"iif", 1}, {"oif", 2},
1828*cf5a6c84SAndroid Build Coastguard Worker {"dev", 3}, {"notify", 4}, {"connected", 5}, {"to", 6}, {NULL, -1}};
1829*cf5a6c84SAndroid Build Coastguard Worker char *idev = NULL, *odev = NULL;
1830*cf5a6c84SAndroid Build Coastguard Worker struct {
1831*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr mhdr;
1832*cf5a6c84SAndroid Build Coastguard Worker struct rtmsg msg;
1833*cf5a6c84SAndroid Build Coastguard Worker char buf[1024];
1834*cf5a6c84SAndroid Build Coastguard Worker } request;
1835*cf5a6c84SAndroid Build Coastguard Worker
1836*cf5a6c84SAndroid Build Coastguard Worker memset(&request, 0, sizeof(request));
1837*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg));
1838*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_flags = NLM_F_REQUEST | NLM_F_ACK;
1839*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_type = RTM_GETROUTE;
1840*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_family = AF_UNSPEC;
1841*cf5a6c84SAndroid Build Coastguard Worker
1842*cf5a6c84SAndroid Build Coastguard Worker for (; *argv; argv++) {
1843*cf5a6c84SAndroid Build Coastguard Worker switch(idx = substring_to_idx(*argv, cmd_objectlist)) {
1844*cf5a6c84SAndroid Build Coastguard Worker case 0: TT.from_ok = 1; // dst address
1845*cf5a6c84SAndroid Build Coastguard Worker case 6: argv++; //fallthrough
1846*cf5a6c84SAndroid Build Coastguard Worker default:
1847*cf5a6c84SAndroid Build Coastguard Worker {
1848*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[8] = {0,}, netmask = 0;
1849*cf5a6c84SAndroid Build Coastguard Worker uint8_t len = 0;
1850*cf5a6c84SAndroid Build Coastguard Worker
1851*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) error_exit("'%s': Missing Prefix", argv[-1]);
1852*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(addr, &netmask, &len, *argv, request.msg.rtm_family);
1853*cf5a6c84SAndroid Build Coastguard Worker if (len) request.msg.rtm_family = ((len == 4) ? AF_INET : AF_INET6);
1854*cf5a6c84SAndroid Build Coastguard Worker netmask = (request.msg.rtm_family == AF_INET6) ? 128 : 32;
1855*cf5a6c84SAndroid Build Coastguard Worker if (!idx) request.msg.rtm_src_len = netmask;
1856*cf5a6c84SAndroid Build Coastguard Worker else request.msg.rtm_dst_len = netmask;
1857*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
1858*cf5a6c84SAndroid Build Coastguard Worker (!idx ? RTA_SRC : RTA_DST), addr, len);
1859*cf5a6c84SAndroid Build Coastguard Worker break;
1860*cf5a6c84SAndroid Build Coastguard Worker }
1861*cf5a6c84SAndroid Build Coastguard Worker case 1:
1862*cf5a6c84SAndroid Build Coastguard Worker case 2:
1863*cf5a6c84SAndroid Build Coastguard Worker case 3:
1864*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
1865*cf5a6c84SAndroid Build Coastguard Worker if (idx == 1) idev = *argv, flag = RTA_IIF;
1866*cf5a6c84SAndroid Build Coastguard Worker else odev = *argv, flag = RTA_OIF;
1867*cf5a6c84SAndroid Build Coastguard Worker idx = get_ifaceindex(*argv, 1);
1868*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
1869*cf5a6c84SAndroid Build Coastguard Worker flag, (char*)&idx, sizeof(idx));
1870*cf5a6c84SAndroid Build Coastguard Worker break;
1871*cf5a6c84SAndroid Build Coastguard Worker case 4:
1872*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_flags |= RTM_F_NOTIFY;
1873*cf5a6c84SAndroid Build Coastguard Worker break;
1874*cf5a6c84SAndroid Build Coastguard Worker case 5:
1875*cf5a6c84SAndroid Build Coastguard Worker TT.connected = 1;
1876*cf5a6c84SAndroid Build Coastguard Worker break;
1877*cf5a6c84SAndroid Build Coastguard Worker }
1878*cf5a6c84SAndroid Build Coastguard Worker }
1879*cf5a6c84SAndroid Build Coastguard Worker if (!request.msg.rtm_dst_len)
1880*cf5a6c84SAndroid Build Coastguard Worker error_exit("need at least destination address");
1881*cf5a6c84SAndroid Build Coastguard Worker
1882*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(0, 0, 0, &request, sizeof(request));
1883*cf5a6c84SAndroid Build Coastguard Worker filter_nlmesg(display_route_info, NULL);
1884*cf5a6c84SAndroid Build Coastguard Worker
1885*cf5a6c84SAndroid Build Coastguard Worker if (TT.connected && !TT.from_ok) {
1886*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr *mhdr = (struct nlmsghdr*)toybuf;
1887*cf5a6c84SAndroid Build Coastguard Worker struct rtmsg *msg = NLMSG_DATA(mhdr);
1888*cf5a6c84SAndroid Build Coastguard Worker int tvar, msglen = mhdr->nlmsg_len - NLMSG_LENGTH(sizeof(struct rtmsg));
1889*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *rta, *attr[RTA_MAX+1] = {0,};
1890*cf5a6c84SAndroid Build Coastguard Worker
1891*cf5a6c84SAndroid Build Coastguard Worker if (mhdr->nlmsg_type != RTM_NEWROUTE) error_exit("not a route?");
1892*cf5a6c84SAndroid Build Coastguard Worker if (msglen < 0) error_exit("wrong len %d", msglen);
1893*cf5a6c84SAndroid Build Coastguard Worker
1894*cf5a6c84SAndroid Build Coastguard Worker tvar = msglen;
1895*cf5a6c84SAndroid Build Coastguard Worker for (rta = RTM_RTA(msg); RTA_OK(rta, tvar); rta=RTA_NEXT(rta, tvar))
1896*cf5a6c84SAndroid Build Coastguard Worker if (rta->rta_type <= RTA_MAX) attr[rta->rta_type] = rta;
1897*cf5a6c84SAndroid Build Coastguard Worker
1898*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_PREFSRC]) {
1899*cf5a6c84SAndroid Build Coastguard Worker attr[RTA_PREFSRC]->rta_type = RTA_SRC;
1900*cf5a6c84SAndroid Build Coastguard Worker msg->rtm_src_len = 8*RTA_PAYLOAD(attr[RTA_PREFSRC]);
1901*cf5a6c84SAndroid Build Coastguard Worker } else if (!attr[RTA_SRC]) error_exit("can't connect the route");
1902*cf5a6c84SAndroid Build Coastguard Worker
1903*cf5a6c84SAndroid Build Coastguard Worker if (!odev && attr[RTA_OIF]) attr[RTA_OIF]->rta_type = 0;
1904*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_GATEWAY]) attr[RTA_GATEWAY]->rta_type = 0;
1905*cf5a6c84SAndroid Build Coastguard Worker if (!idev && attr[RTA_IIF]) attr[RTA_IIF]->rta_type = 0;
1906*cf5a6c84SAndroid Build Coastguard Worker mhdr->nlmsg_flags = NLM_F_REQUEST | NLM_F_ACK;
1907*cf5a6c84SAndroid Build Coastguard Worker mhdr->nlmsg_type = RTM_GETROUTE;
1908*cf5a6c84SAndroid Build Coastguard Worker mhdr->nlmsg_pid = 0;
1909*cf5a6c84SAndroid Build Coastguard Worker xclose(TT.sockfd);
1910*cf5a6c84SAndroid Build Coastguard Worker TT.sockfd = xsocket(AF_NETLINK, SOCK_RAW, NETLINK_ROUTE);
1911*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(0, 0, 0, mhdr, mhdr->nlmsg_len);
1912*cf5a6c84SAndroid Build Coastguard Worker filter_nlmesg(display_route_info, NULL);
1913*cf5a6c84SAndroid Build Coastguard Worker }
1914*cf5a6c84SAndroid Build Coastguard Worker return 0;
1915*cf5a6c84SAndroid Build Coastguard Worker }
1916*cf5a6c84SAndroid Build Coastguard Worker
route_show_flush(char ** argv)1917*cf5a6c84SAndroid Build Coastguard Worker static int route_show_flush(char **argv)
1918*cf5a6c84SAndroid Build Coastguard Worker {
1919*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist[] = {{"protocol", 0}, {"dev", 1}, {"oif", 2},
1920*cf5a6c84SAndroid Build Coastguard Worker {"iif", 3}, {"via", 4}, {"table", 5}, {"cache", 6}, {"from", 7},
1921*cf5a6c84SAndroid Build Coastguard Worker {"to", 8}, {"all", 9}, {"root", 10}, {"match", 11}, {"exact", 12},
1922*cf5a6c84SAndroid Build Coastguard Worker {"main", 13}, {NULL,-1}};
1923*cf5a6c84SAndroid Build Coastguard Worker int family = TT.addressfamily, idx;
1924*cf5a6c84SAndroid Build Coastguard Worker struct {
1925*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr mhdr;
1926*cf5a6c84SAndroid Build Coastguard Worker struct rtmsg msg;
1927*cf5a6c84SAndroid Build Coastguard Worker } request;
1928*cf5a6c84SAndroid Build Coastguard Worker
1929*cf5a6c84SAndroid Build Coastguard Worker if (*argv[-1] == 'f') TT.flush = 1;
1930*cf5a6c84SAndroid Build Coastguard Worker if (TT.flush && !*argv) show_iproute_help();
1931*cf5a6c84SAndroid Build Coastguard Worker
1932*cf5a6c84SAndroid Build Coastguard Worker gfilter.tb = RT_TABLE_MAIN;
1933*cf5a6c84SAndroid Build Coastguard Worker for (; *argv; argv++) {
1934*cf5a6c84SAndroid Build Coastguard Worker switch (idx = substring_to_idx(*argv, cmd_objectlist)) {
1935*cf5a6c84SAndroid Build Coastguard Worker case 0:
1936*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
1937*cf5a6c84SAndroid Build Coastguard Worker if ((idx = idxfromRPDB(*argv,RPDB_rtprotos)) < 0)
1938*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid argument protocol.");
1939*cf5a6c84SAndroid Build Coastguard Worker gfilter.proto = idx;
1940*cf5a6c84SAndroid Build Coastguard Worker break;
1941*cf5a6c84SAndroid Build Coastguard Worker case 1:
1942*cf5a6c84SAndroid Build Coastguard Worker case 2:
1943*cf5a6c84SAndroid Build Coastguard Worker case 3:
1944*cf5a6c84SAndroid Build Coastguard Worker {
1945*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
1946*cf5a6c84SAndroid Build Coastguard Worker int dev = get_ifaceindex(*argv, 1);
1947*cf5a6c84SAndroid Build Coastguard Worker
1948*cf5a6c84SAndroid Build Coastguard Worker if (idx == 3) gfilter.idev = dev;
1949*cf5a6c84SAndroid Build Coastguard Worker else gfilter.odev = dev;
1950*cf5a6c84SAndroid Build Coastguard Worker }
1951*cf5a6c84SAndroid Build Coastguard Worker break;
1952*cf5a6c84SAndroid Build Coastguard Worker case 4:
1953*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
1954*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(gfilter.rvia.addr, &gfilter.rvia.netmask,
1955*cf5a6c84SAndroid Build Coastguard Worker &gfilter.rvia.len, *argv, gfilter.rvia.family);
1956*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.rvia.len)
1957*cf5a6c84SAndroid Build Coastguard Worker gfilter.rvia.family = ((gfilter.rvia.len == 4) ?
1958*cf5a6c84SAndroid Build Coastguard Worker AF_INET : AF_INET6);
1959*cf5a6c84SAndroid Build Coastguard Worker break;
1960*cf5a6c84SAndroid Build Coastguard Worker case 5:
1961*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
1962*cf5a6c84SAndroid Build Coastguard Worker idx = substring_to_idx(*argv, cmd_objectlist);
1963*cf5a6c84SAndroid Build Coastguard Worker if (idx == 6) gfilter.tb = -1;
1964*cf5a6c84SAndroid Build Coastguard Worker else if (idx == 9) gfilter.tb = 0;
1965*cf5a6c84SAndroid Build Coastguard Worker else if (idx != 13) {
1966*cf5a6c84SAndroid Build Coastguard Worker if ((gfilter.tb = idxfromRPDB(*argv, RPDB_rttables)) < 0)
1967*cf5a6c84SAndroid Build Coastguard Worker error_exit("table %s is invalid.", *argv);
1968*cf5a6c84SAndroid Build Coastguard Worker }
1969*cf5a6c84SAndroid Build Coastguard Worker break;
1970*cf5a6c84SAndroid Build Coastguard Worker case 6:
1971*cf5a6c84SAndroid Build Coastguard Worker gfilter.tb = -1;
1972*cf5a6c84SAndroid Build Coastguard Worker break;
1973*cf5a6c84SAndroid Build Coastguard Worker case 7:
1974*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
1975*cf5a6c84SAndroid Build Coastguard Worker idx = substring_to_idx(*argv, cmd_objectlist);
1976*cf5a6c84SAndroid Build Coastguard Worker if (idx < 0) if (!*++argv) show_iproute_help();
1977*cf5a6c84SAndroid Build Coastguard Worker if (idx == 10)
1978*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
1979*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(gfilter.rsrc.addr, &gfilter.rsrc.netmask,
1980*cf5a6c84SAndroid Build Coastguard Worker &gfilter.rsrc.len, *argv, gfilter.rsrc.family);
1981*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.rsrc.len)
1982*cf5a6c84SAndroid Build Coastguard Worker gfilter.rsrc.family = ((gfilter.rsrc.len == 4) ?
1983*cf5a6c84SAndroid Build Coastguard Worker AF_INET : AF_INET6);
1984*cf5a6c84SAndroid Build Coastguard Worker else {
1985*cf5a6c84SAndroid Build Coastguard Worker if ((idx == 12 ||idx == 11) && !*++argv) show_iproute_help();
1986*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(gfilter.msrc.addr, &gfilter.msrc.netmask,
1987*cf5a6c84SAndroid Build Coastguard Worker &gfilter.msrc.len, *argv, gfilter.msrc.family);
1988*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.msrc.len)
1989*cf5a6c84SAndroid Build Coastguard Worker gfilter.msrc.family = ((gfilter.msrc.len == 4) ?
1990*cf5a6c84SAndroid Build Coastguard Worker AF_INET : AF_INET6);
1991*cf5a6c84SAndroid Build Coastguard Worker if (idx != 11) gfilter.rsrc = gfilter.msrc;
1992*cf5a6c84SAndroid Build Coastguard Worker }
1993*cf5a6c84SAndroid Build Coastguard Worker break;
1994*cf5a6c84SAndroid Build Coastguard Worker case 8:
1995*cf5a6c84SAndroid Build Coastguard Worker idx = substring_to_idx(*argv, cmd_objectlist);
1996*cf5a6c84SAndroid Build Coastguard Worker if (idx != -1 && !*++argv) show_iproute_help();
1997*cf5a6c84SAndroid Build Coastguard Worker default: // fallthrough
1998*cf5a6c84SAndroid Build Coastguard Worker if (idx == 10) {
1999*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2000*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(gfilter.rdst.addr, &gfilter.rdst.netmask,
2001*cf5a6c84SAndroid Build Coastguard Worker &gfilter.rdst.len, *argv, gfilter.rdst.family);
2002*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.rdst.len)
2003*cf5a6c84SAndroid Build Coastguard Worker gfilter.rdst.family = ((gfilter.rdst.len == 4) ?
2004*cf5a6c84SAndroid Build Coastguard Worker AF_INET : AF_INET6);
2005*cf5a6c84SAndroid Build Coastguard Worker }
2006*cf5a6c84SAndroid Build Coastguard Worker else {
2007*cf5a6c84SAndroid Build Coastguard Worker if ((idx == 12 ||idx == 11) && !*++argv) show_iproute_help();
2008*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(gfilter.mdst.addr, &gfilter.mdst.netmask,
2009*cf5a6c84SAndroid Build Coastguard Worker &gfilter.mdst.len, *argv, gfilter.mdst.family);
2010*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.mdst.len)
2011*cf5a6c84SAndroid Build Coastguard Worker gfilter.mdst.family = ((gfilter.mdst.len == 4) ?
2012*cf5a6c84SAndroid Build Coastguard Worker AF_INET : AF_INET6);
2013*cf5a6c84SAndroid Build Coastguard Worker if (idx != 11) gfilter.rdst = gfilter.mdst;
2014*cf5a6c84SAndroid Build Coastguard Worker }
2015*cf5a6c84SAndroid Build Coastguard Worker break;
2016*cf5a6c84SAndroid Build Coastguard Worker }
2017*cf5a6c84SAndroid Build Coastguard Worker }
2018*cf5a6c84SAndroid Build Coastguard Worker if (family == AF_UNSPEC && gfilter.tb) family = AF_INET;
2019*cf5a6c84SAndroid Build Coastguard Worker
2020*cf5a6c84SAndroid Build Coastguard Worker if (TT.flush) {
2021*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.tb < 0) { // flush table cache
2022*cf5a6c84SAndroid Build Coastguard Worker if (family != AF_INET6) {
2023*cf5a6c84SAndroid Build Coastguard Worker FILE *fp = xfopen("/proc/sys/net/ipv4/route/flush", "w");
2024*cf5a6c84SAndroid Build Coastguard Worker
2025*cf5a6c84SAndroid Build Coastguard Worker if (fwrite("-1",1,2,fp) < 2) error_exit("can't flush routing cache");
2026*cf5a6c84SAndroid Build Coastguard Worker fclose(fp);
2027*cf5a6c84SAndroid Build Coastguard Worker }
2028*cf5a6c84SAndroid Build Coastguard Worker if (family == AF_INET) return 0;
2029*cf5a6c84SAndroid Build Coastguard Worker }
2030*cf5a6c84SAndroid Build Coastguard Worker }
2031*cf5a6c84SAndroid Build Coastguard Worker
2032*cf5a6c84SAndroid Build Coastguard Worker memset(&request, 0, sizeof (request));
2033*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_len = NLMSG_LENGTH(sizeof (struct rtmsg));
2034*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_flags = NLM_F_REQUEST;
2035*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_flags |= NLM_F_ROOT | NLM_F_MATCH;
2036*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_type = RTM_GETROUTE;
2037*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_family = family;
2038*cf5a6c84SAndroid Build Coastguard Worker if (gfilter.tb < 0) request.msg.rtm_flags = RTM_F_CLONED;
2039*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(0, 0, 0, (void*)&request, sizeof (request));
2040*cf5a6c84SAndroid Build Coastguard Worker return (filter_nlmesg(display_route_info, NULL));
2041*cf5a6c84SAndroid Build Coastguard Worker }
2042*cf5a6c84SAndroid Build Coastguard Worker
route_update(char ** argv,unsigned int route_flags)2043*cf5a6c84SAndroid Build Coastguard Worker static int route_update(char **argv, unsigned int route_flags)
2044*cf5a6c84SAndroid Build Coastguard Worker {
2045*cf5a6c84SAndroid Build Coastguard Worker char mxbuf[256], *d = NULL;
2046*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *mxrta = (void*)mxbuf;
2047*cf5a6c84SAndroid Build Coastguard Worker unsigned mxlock = 0, ok = 0;
2048*cf5a6c84SAndroid Build Coastguard Worker int idx;
2049*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[8] = {0,}, netmask = 0;
2050*cf5a6c84SAndroid Build Coastguard Worker uint8_t len = 0;
2051*cf5a6c84SAndroid Build Coastguard Worker
2052*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist[] = {{"src", 0}, {"via", 1}, {"mtu", 2},
2053*cf5a6c84SAndroid Build Coastguard Worker {"lock", 3}, {"protocol", 4}, {"table", 5}, {"dev", 6}, {"oif", 7},
2054*cf5a6c84SAndroid Build Coastguard Worker {"to", 8}, {"metric", 9}, {NULL,-1}
2055*cf5a6c84SAndroid Build Coastguard Worker };
2056*cf5a6c84SAndroid Build Coastguard Worker enum {
2057*cf5a6c84SAndroid Build Coastguard Worker gtwy_ok = 1,
2058*cf5a6c84SAndroid Build Coastguard Worker dst_ok = 2,
2059*cf5a6c84SAndroid Build Coastguard Worker proto_ok = 4,
2060*cf5a6c84SAndroid Build Coastguard Worker type_ok = 8
2061*cf5a6c84SAndroid Build Coastguard Worker };
2062*cf5a6c84SAndroid Build Coastguard Worker struct {
2063*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr hdr;
2064*cf5a6c84SAndroid Build Coastguard Worker struct rtmsg msg;
2065*cf5a6c84SAndroid Build Coastguard Worker char buf[1024];
2066*cf5a6c84SAndroid Build Coastguard Worker } req;
2067*cf5a6c84SAndroid Build Coastguard Worker
2068*cf5a6c84SAndroid Build Coastguard Worker memset(&req, 0, sizeof(req));
2069*cf5a6c84SAndroid Build Coastguard Worker req.hdr.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg));
2070*cf5a6c84SAndroid Build Coastguard Worker req.hdr.nlmsg_flags = NLM_F_ACK| NLM_F_REQUEST | route_flags;
2071*cf5a6c84SAndroid Build Coastguard Worker req.hdr.nlmsg_type = TT.route_cmd;
2072*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_family = AF_UNSPEC;
2073*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_table = RT_TABLE_MAIN;
2074*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_scope = RT_SCOPE_NOWHERE;
2075*cf5a6c84SAndroid Build Coastguard Worker
2076*cf5a6c84SAndroid Build Coastguard Worker if (TT.route_cmd != RTM_DELROUTE) {
2077*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_protocol = RTPROT_BOOT;
2078*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_scope = RT_SCOPE_UNIVERSE;
2079*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_type = RTN_UNICAST;
2080*cf5a6c84SAndroid Build Coastguard Worker }
2081*cf5a6c84SAndroid Build Coastguard Worker
2082*cf5a6c84SAndroid Build Coastguard Worker mxrta->rta_type = RTA_METRICS;
2083*cf5a6c84SAndroid Build Coastguard Worker mxrta->rta_len = RTA_LENGTH(0);
2084*cf5a6c84SAndroid Build Coastguard Worker
2085*cf5a6c84SAndroid Build Coastguard Worker for (; *argv; argv++) {
2086*cf5a6c84SAndroid Build Coastguard Worker idx = substring_to_idx(*argv, cmd_objectlist);
2087*cf5a6c84SAndroid Build Coastguard Worker if (!idx) {
2088*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2089*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(addr, &netmask, &len, *argv, req.msg.rtm_family);
2090*cf5a6c84SAndroid Build Coastguard Worker if (len) req.msg.rtm_family = ((len == 4) ? AF_INET : AF_INET6);
2091*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.hdr, sizeof(req), RTA_PREFSRC, addr, len);
2092*cf5a6c84SAndroid Build Coastguard Worker } else if (idx == 1) {
2093*cf5a6c84SAndroid Build Coastguard Worker ok |= gtwy_ok;
2094*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2095*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(addr, &netmask, &len, *argv, req.msg.rtm_family);
2096*cf5a6c84SAndroid Build Coastguard Worker if (len) req.msg.rtm_family = ((len == 4) ? AF_INET : AF_INET6);
2097*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.hdr, sizeof(req),RTA_GATEWAY, addr, len);
2098*cf5a6c84SAndroid Build Coastguard Worker } else if (idx == 2) {
2099*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2100*cf5a6c84SAndroid Build Coastguard Worker if (substring_to_idx(*argv, cmd_objectlist ) == 3) {
2101*cf5a6c84SAndroid Build Coastguard Worker mxlock |= (1 << RTAX_MTU);
2102*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2103*cf5a6c84SAndroid Build Coastguard Worker }
2104*cf5a6c84SAndroid Build Coastguard Worker idx = atolx(*argv);
2105*cf5a6c84SAndroid Build Coastguard Worker add_uint32_rtattr_to_buffer(mxrta, sizeof(mxbuf), RTAX_MTU, idx);
2106*cf5a6c84SAndroid Build Coastguard Worker } else if (idx == 4) {
2107*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2108*cf5a6c84SAndroid Build Coastguard Worker if ((idx = idxfromRPDB(*argv,RPDB_rtprotos)) < 0)
2109*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid argument protocol %s.",*argv);
2110*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_protocol = idx;
2111*cf5a6c84SAndroid Build Coastguard Worker ok |= proto_ok;
2112*cf5a6c84SAndroid Build Coastguard Worker } else if (idx == 5) {
2113*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2114*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_table = idxfromRPDB(*argv, RPDB_rttables);
2115*cf5a6c84SAndroid Build Coastguard Worker } else if (idx == 6 || idx == 7) {
2116*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2117*cf5a6c84SAndroid Build Coastguard Worker d = *argv;
2118*cf5a6c84SAndroid Build Coastguard Worker } else if (idx == 9) {
2119*cf5a6c84SAndroid Build Coastguard Worker unsigned long metric;
2120*cf5a6c84SAndroid Build Coastguard Worker unsigned int res;
2121*cf5a6c84SAndroid Build Coastguard Worker char* ptr;
2122*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2123*cf5a6c84SAndroid Build Coastguard Worker metric = strtoul(*argv, &ptr, 0);
2124*cf5a6c84SAndroid Build Coastguard Worker if (!(!*ptr && metric <= 0xFFFFFFFFUL))
2125*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid argument metric %s.",*argv);
2126*cf5a6c84SAndroid Build Coastguard Worker else
2127*cf5a6c84SAndroid Build Coastguard Worker res = metric;
2128*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.hdr, sizeof(req),
2129*cf5a6c84SAndroid Build Coastguard Worker RTA_PRIORITY, (char*)&res, sizeof(res));
2130*cf5a6c84SAndroid Build Coastguard Worker } else {
2131*cf5a6c84SAndroid Build Coastguard Worker if (idx == 8)
2132*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2133*cf5a6c84SAndroid Build Coastguard Worker idx = substring_to_idx(*argv,rtmtypes);
2134*cf5a6c84SAndroid Build Coastguard Worker if (idx != -1) {
2135*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) show_iproute_help();
2136*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_type = idx;
2137*cf5a6c84SAndroid Build Coastguard Worker ok |= type_ok;
2138*cf5a6c84SAndroid Build Coastguard Worker }
2139*cf5a6c84SAndroid Build Coastguard Worker if (ok & dst_ok) error_exit("Duplicate argument 'to'");
2140*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(addr, &netmask, &len, *argv, req.msg.rtm_family);
2141*cf5a6c84SAndroid Build Coastguard Worker if (len) req.msg.rtm_family = ((len == 4) ? AF_INET : AF_INET6);
2142*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_dst_len = netmask;
2143*cf5a6c84SAndroid Build Coastguard Worker ok |= dst_ok;
2144*cf5a6c84SAndroid Build Coastguard Worker if (len) add_string_to_rtattr(&req.hdr, sizeof(req),RTA_DST, addr, len);
2145*cf5a6c84SAndroid Build Coastguard Worker }
2146*cf5a6c84SAndroid Build Coastguard Worker }
2147*cf5a6c84SAndroid Build Coastguard Worker
2148*cf5a6c84SAndroid Build Coastguard Worker if (d) {
2149*cf5a6c84SAndroid Build Coastguard Worker idx = get_ifaceindex(d,1);
2150*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.hdr, sizeof(req),
2151*cf5a6c84SAndroid Build Coastguard Worker RTA_OIF, (char*)&idx, sizeof(idx));
2152*cf5a6c84SAndroid Build Coastguard Worker }
2153*cf5a6c84SAndroid Build Coastguard Worker if (mxrta->rta_len > RTA_LENGTH(0)) {
2154*cf5a6c84SAndroid Build Coastguard Worker if (mxlock)
2155*cf5a6c84SAndroid Build Coastguard Worker add_uint32_rtattr_to_buffer(mxrta, sizeof(mxbuf), RTAX_LOCK, mxlock);
2156*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&req.hdr, sizeof(req),
2157*cf5a6c84SAndroid Build Coastguard Worker RTA_METRICS, RTA_DATA(mxrta), RTA_PAYLOAD(mxrta));
2158*cf5a6c84SAndroid Build Coastguard Worker }
2159*cf5a6c84SAndroid Build Coastguard Worker
2160*cf5a6c84SAndroid Build Coastguard Worker if (req.msg.rtm_type == RTN_LOCAL || req.msg.rtm_type == RTN_NAT)
2161*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_scope = RT_SCOPE_HOST;
2162*cf5a6c84SAndroid Build Coastguard Worker else if (req.msg.rtm_type == RTN_BROADCAST||req.msg.rtm_type == RTN_MULTICAST
2163*cf5a6c84SAndroid Build Coastguard Worker || req.msg.rtm_type == RTN_ANYCAST)
2164*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_scope = RT_SCOPE_LINK;
2165*cf5a6c84SAndroid Build Coastguard Worker else if (req.msg.rtm_type == RTN_UNICAST || req.msg.rtm_type == RTN_UNSPEC) {
2166*cf5a6c84SAndroid Build Coastguard Worker if (TT.route_cmd == RTM_DELROUTE)
2167*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_scope = RT_SCOPE_NOWHERE;
2168*cf5a6c84SAndroid Build Coastguard Worker else if (!(ok & gtwy_ok))
2169*cf5a6c84SAndroid Build Coastguard Worker req.msg.rtm_scope = RT_SCOPE_LINK;
2170*cf5a6c84SAndroid Build Coastguard Worker }
2171*cf5a6c84SAndroid Build Coastguard Worker if (req.msg.rtm_family == AF_UNSPEC) req.msg.rtm_family = AF_INET;
2172*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(0, 0, 0, &req, sizeof(req));
2173*cf5a6c84SAndroid Build Coastguard Worker filter_nlmesg(NULL, NULL);
2174*cf5a6c84SAndroid Build Coastguard Worker return 0;
2175*cf5a6c84SAndroid Build Coastguard Worker }
2176*cf5a6c84SAndroid Build Coastguard Worker
iproute(char ** argv)2177*cf5a6c84SAndroid Build Coastguard Worker static int iproute(char **argv)
2178*cf5a6c84SAndroid Build Coastguard Worker {
2179*cf5a6c84SAndroid Build Coastguard Worker int idx = 1;
2180*cf5a6c84SAndroid Build Coastguard Worker struct arglist cmd_objectlist1[] = {{"add", 0}, {"append", 1},{"change", 2},
2181*cf5a6c84SAndroid Build Coastguard Worker {"chg", 3},{"delete",4}, {"get", 5}, {"list", 6}, {"show", 7},
2182*cf5a6c84SAndroid Build Coastguard Worker {"prepend", 8},{"replace", 9},{"test", 10}, {"flush", 11},{NULL,-1}};
2183*cf5a6c84SAndroid Build Coastguard Worker
2184*cf5a6c84SAndroid Build Coastguard Worker TT.route_cmd = RTM_NEWROUTE;
2185*cf5a6c84SAndroid Build Coastguard Worker switch (idx = substring_to_idx(*argv , cmd_objectlist1)) {
2186*cf5a6c84SAndroid Build Coastguard Worker case 0: // add
2187*cf5a6c84SAndroid Build Coastguard Worker return route_update(++argv , NLM_F_CREATE|NLM_F_EXCL);
2188*cf5a6c84SAndroid Build Coastguard Worker case 1: // append
2189*cf5a6c84SAndroid Build Coastguard Worker return route_update(++argv , NLM_F_CREATE|NLM_F_APPEND);
2190*cf5a6c84SAndroid Build Coastguard Worker case 2: // change
2191*cf5a6c84SAndroid Build Coastguard Worker case 3: // chg
2192*cf5a6c84SAndroid Build Coastguard Worker return route_update(++argv , NLM_F_REPLACE);
2193*cf5a6c84SAndroid Build Coastguard Worker case 4: // delete
2194*cf5a6c84SAndroid Build Coastguard Worker TT.route_cmd = RTM_DELROUTE;
2195*cf5a6c84SAndroid Build Coastguard Worker return route_update(++argv , RTM_DELROUTE);
2196*cf5a6c84SAndroid Build Coastguard Worker case 5:
2197*cf5a6c84SAndroid Build Coastguard Worker return route_get(++argv);
2198*cf5a6c84SAndroid Build Coastguard Worker case 6:
2199*cf5a6c84SAndroid Build Coastguard Worker case 7:
2200*cf5a6c84SAndroid Build Coastguard Worker return route_show_flush(++argv);
2201*cf5a6c84SAndroid Build Coastguard Worker case 8: // prepend
2202*cf5a6c84SAndroid Build Coastguard Worker return route_update(++argv , NLM_F_CREATE);
2203*cf5a6c84SAndroid Build Coastguard Worker case 9: // replace
2204*cf5a6c84SAndroid Build Coastguard Worker return route_update(++argv , NLM_F_CREATE|NLM_F_REPLACE);
2205*cf5a6c84SAndroid Build Coastguard Worker case 10: // test
2206*cf5a6c84SAndroid Build Coastguard Worker return route_update(++argv , NLM_F_EXCL);
2207*cf5a6c84SAndroid Build Coastguard Worker case 11: // flush
2208*cf5a6c84SAndroid Build Coastguard Worker return route_show_flush(++argv);
2209*cf5a6c84SAndroid Build Coastguard Worker default:
2210*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) return route_show_flush(argv);
2211*cf5a6c84SAndroid Build Coastguard Worker else show_iproute_help();
2212*cf5a6c84SAndroid Build Coastguard Worker }
2213*cf5a6c84SAndroid Build Coastguard Worker return 0; // non reachable code.
2214*cf5a6c84SAndroid Build Coastguard Worker }
2215*cf5a6c84SAndroid Build Coastguard Worker
2216*cf5a6c84SAndroid Build Coastguard Worker
2217*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
2218*cf5a6c84SAndroid Build Coastguard Worker // code for ip rule.
2219*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
show_iprule_help(void)2220*cf5a6c84SAndroid Build Coastguard Worker static void show_iprule_help(void)
2221*cf5a6c84SAndroid Build Coastguard Worker {
2222*cf5a6c84SAndroid Build Coastguard Worker error_exit("usage: ip rule [ list | add | del ] SELECTOR ACTION\n"
2223*cf5a6c84SAndroid Build Coastguard Worker "SELECTOR := [ from PREFIX ] [ to PREFIX ] [pref NUMBER] [ tos TOS ]\n"
2224*cf5a6c84SAndroid Build Coastguard Worker " [ fwmark FWMARK] [ dev/iif STRING ] [type TYPE]\n"
2225*cf5a6c84SAndroid Build Coastguard Worker "ACTION := [ table TABLE_ID ] [ realms [SRCREALM/]DSTREALM ]");
2226*cf5a6c84SAndroid Build Coastguard Worker }
2227*cf5a6c84SAndroid Build Coastguard Worker
ruleupdate(char ** argv)2228*cf5a6c84SAndroid Build Coastguard Worker static int ruleupdate(char **argv)
2229*cf5a6c84SAndroid Build Coastguard Worker {
2230*cf5a6c84SAndroid Build Coastguard Worker int8_t idx, tflag = 0, opt = (*argv[-1] == 'a') ? RTM_NEWRULE : RTM_DELRULE;
2231*cf5a6c84SAndroid Build Coastguard Worker struct arglist options[] = {{"from", 0}, {"to", 1}, {"preference", 2},
2232*cf5a6c84SAndroid Build Coastguard Worker {"order", 2}, {"priority", 2}, {"tos", 3}, {"dsfield", 3}, {"fwmark", 4},
2233*cf5a6c84SAndroid Build Coastguard Worker {"realms", 5}, {"table", 6}, {"lookup", 6}, {"dev", 7}, {"iif", 7},
2234*cf5a6c84SAndroid Build Coastguard Worker {"nat", 8}, {"map-to", 8}, {"type", 9}, {"help", 10}, {NULL, -1}};
2235*cf5a6c84SAndroid Build Coastguard Worker struct {
2236*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr mhdr;
2237*cf5a6c84SAndroid Build Coastguard Worker struct rtmsg msg;
2238*cf5a6c84SAndroid Build Coastguard Worker char buf[1024];
2239*cf5a6c84SAndroid Build Coastguard Worker } request;
2240*cf5a6c84SAndroid Build Coastguard Worker
2241*cf5a6c84SAndroid Build Coastguard Worker memset(&request, 0, sizeof(request));
2242*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_type = opt;
2243*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg));
2244*cf5a6c84SAndroid Build Coastguard Worker request.mhdr.nlmsg_flags = NLM_F_REQUEST | NLM_F_ACK |
2245*cf5a6c84SAndroid Build Coastguard Worker ((opt == RTM_DELRULE) ? 0 : NLM_F_CREATE | NLM_F_EXCL);
2246*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_family = TT.addressfamily;
2247*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_protocol = RTPROT_BOOT;
2248*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_scope = RT_SCOPE_UNIVERSE;
2249*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_table = 0;
2250*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_type = ((opt == RTM_DELRULE) ? RTN_UNSPEC : RTN_UNICAST);
2251*cf5a6c84SAndroid Build Coastguard Worker
2252*cf5a6c84SAndroid Build Coastguard Worker for (; *argv; argv++) {
2253*cf5a6c84SAndroid Build Coastguard Worker switch ((idx = substring_to_idx(*argv, options))) {
2254*cf5a6c84SAndroid Build Coastguard Worker case 0:
2255*cf5a6c84SAndroid Build Coastguard Worker case 1:
2256*cf5a6c84SAndroid Build Coastguard Worker { // e.g. from IP/Netmask and to IP/Netmask.
2257*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[4] = {0,}, netmask = 0;
2258*cf5a6c84SAndroid Build Coastguard Worker uint8_t len = 0, *tmp;
2259*cf5a6c84SAndroid Build Coastguard Worker
2260*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("'%s': Missing Prefix", argv[-1]);
2261*cf5a6c84SAndroid Build Coastguard Worker parse_prefix(addr, &netmask, &len, *argv, request.msg.rtm_family);
2262*cf5a6c84SAndroid Build Coastguard Worker
2263*cf5a6c84SAndroid Build Coastguard Worker tmp = idx ? &request.msg.rtm_dst_len : &request.msg.rtm_src_len;
2264*cf5a6c84SAndroid Build Coastguard Worker if (!netmask) *tmp = 0;
2265*cf5a6c84SAndroid Build Coastguard Worker else *tmp = netmask;
2266*cf5a6c84SAndroid Build Coastguard Worker
2267*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
2268*cf5a6c84SAndroid Build Coastguard Worker (idx ? RTA_DST : RTA_SRC), addr, len);
2269*cf5a6c84SAndroid Build Coastguard Worker }
2270*cf5a6c84SAndroid Build Coastguard Worker break;
2271*cf5a6c84SAndroid Build Coastguard Worker case 2:
2272*cf5a6c84SAndroid Build Coastguard Worker case 4:
2273*cf5a6c84SAndroid Build Coastguard Worker { // e.g. Preference p# and fwmark MARK
2274*cf5a6c84SAndroid Build Coastguard Worker uint32_t pref;
2275*cf5a6c84SAndroid Build Coastguard Worker char *ptr;
2276*cf5a6c84SAndroid Build Coastguard Worker
2277*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv)
2278*cf5a6c84SAndroid Build Coastguard Worker error_exit("Missing %s", (idx == 2) ? "Preference" : "fwmark");
2279*cf5a6c84SAndroid Build Coastguard Worker pref = strtoul(*argv, &ptr, 0);
2280*cf5a6c84SAndroid Build Coastguard Worker if (!ptr || (ptr == *argv) || *ptr || pref > 0xFFFFFFFFUL)
2281*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid %s", (idx == 2) ? "Preference" : "fwmark");
2282*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
2283*cf5a6c84SAndroid Build Coastguard Worker ((idx == 2) ? RTA_PRIORITY : RTA_PROTOINFO),
2284*cf5a6c84SAndroid Build Coastguard Worker (void *)&pref, sizeof(uint32_t));
2285*cf5a6c84SAndroid Build Coastguard Worker }
2286*cf5a6c84SAndroid Build Coastguard Worker break;
2287*cf5a6c84SAndroid Build Coastguard Worker case 3:
2288*cf5a6c84SAndroid Build Coastguard Worker {
2289*cf5a6c84SAndroid Build Coastguard Worker uint32_t tos;
2290*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Missing TOS key");
2291*cf5a6c84SAndroid Build Coastguard Worker if ((tos = idxfromRPDB(*argv, RPDB_rtdsfield)) < 0)
2292*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid TOS");
2293*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_tos = tos;
2294*cf5a6c84SAndroid Build Coastguard Worker }
2295*cf5a6c84SAndroid Build Coastguard Worker break;
2296*cf5a6c84SAndroid Build Coastguard Worker case 5:
2297*cf5a6c84SAndroid Build Coastguard Worker { // e.g. realms FROM_realm/TO_realm
2298*cf5a6c84SAndroid Build Coastguard Worker uint32_t realms = 0;
2299*cf5a6c84SAndroid Build Coastguard Worker int ret;
2300*cf5a6c84SAndroid Build Coastguard Worker char *ptr;
2301*cf5a6c84SAndroid Build Coastguard Worker
2302*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Missing REALMSID");
2303*cf5a6c84SAndroid Build Coastguard Worker if ((ptr = strchr(*argv, '/'))) {
2304*cf5a6c84SAndroid Build Coastguard Worker *ptr = 0;
2305*cf5a6c84SAndroid Build Coastguard Worker if ((ret = idxfromRPDB(*argv, RPDB_rtrealms)) < 0)
2306*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid realms");
2307*cf5a6c84SAndroid Build Coastguard Worker realms = ret;
2308*cf5a6c84SAndroid Build Coastguard Worker realms <<= 16;
2309*cf5a6c84SAndroid Build Coastguard Worker *ptr++ = '/';
2310*cf5a6c84SAndroid Build Coastguard Worker } else ptr = *argv;
2311*cf5a6c84SAndroid Build Coastguard Worker if ((ret = idxfromRPDB(ptr, RPDB_rtrealms)) < 0)
2312*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid realms");
2313*cf5a6c84SAndroid Build Coastguard Worker realms |= ret;
2314*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
2315*cf5a6c84SAndroid Build Coastguard Worker RTA_FLOW, (void *)&realms, sizeof(uint32_t));
2316*cf5a6c84SAndroid Build Coastguard Worker }
2317*cf5a6c84SAndroid Build Coastguard Worker break;
2318*cf5a6c84SAndroid Build Coastguard Worker case 6:
2319*cf5a6c84SAndroid Build Coastguard Worker { // e.g. table tid/tableName
2320*cf5a6c84SAndroid Build Coastguard Worker int tid;
2321*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Missing TableID");
2322*cf5a6c84SAndroid Build Coastguard Worker if ((tid = idxfromRPDB(*argv, RPDB_rttables)) < 0)
2323*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid TID");
2324*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_table = tid;
2325*cf5a6c84SAndroid Build Coastguard Worker tflag = 1;
2326*cf5a6c84SAndroid Build Coastguard Worker }
2327*cf5a6c84SAndroid Build Coastguard Worker break;
2328*cf5a6c84SAndroid Build Coastguard Worker case 7:
2329*cf5a6c84SAndroid Build Coastguard Worker {
2330*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Missing dev/iif NAME");
2331*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
2332*cf5a6c84SAndroid Build Coastguard Worker RTA_IIF, *argv, strlen(*argv)+1);
2333*cf5a6c84SAndroid Build Coastguard Worker }
2334*cf5a6c84SAndroid Build Coastguard Worker break;
2335*cf5a6c84SAndroid Build Coastguard Worker case 8:
2336*cf5a6c84SAndroid Build Coastguard Worker {
2337*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr[4] = {0,};
2338*cf5a6c84SAndroid Build Coastguard Worker uint8_t af = AF_UNSPEC;
2339*cf5a6c84SAndroid Build Coastguard Worker
2340*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("Missing nat/map-to ADDRESS");
2341*cf5a6c84SAndroid Build Coastguard Worker if (get_prefix(addr, &af /* Un-used variable */, *argv, AF_INET))
2342*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid mapping Address");
2343*cf5a6c84SAndroid Build Coastguard Worker
2344*cf5a6c84SAndroid Build Coastguard Worker add_string_to_rtattr(&request.mhdr, sizeof(request),
2345*cf5a6c84SAndroid Build Coastguard Worker RTA_GATEWAY, addr, sizeof(uint32_t));
2346*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_type = RTN_NAT;
2347*cf5a6c84SAndroid Build Coastguard Worker }
2348*cf5a6c84SAndroid Build Coastguard Worker break;
2349*cf5a6c84SAndroid Build Coastguard Worker case 9:
2350*cf5a6c84SAndroid Build Coastguard Worker {
2351*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("TYPE Missing");
2352*cf5a6c84SAndroid Build Coastguard Worker request.msg.rtm_type = rtmtype_str2idx(*argv);
2353*cf5a6c84SAndroid Build Coastguard Worker }
2354*cf5a6c84SAndroid Build Coastguard Worker break;
2355*cf5a6c84SAndroid Build Coastguard Worker case 10:
2356*cf5a6c84SAndroid Build Coastguard Worker show_iprule_help();
2357*cf5a6c84SAndroid Build Coastguard Worker break; // Unreachable code.
2358*cf5a6c84SAndroid Build Coastguard Worker default:
2359*cf5a6c84SAndroid Build Coastguard Worker error_exit("Invalid argument '%s'", *argv);
2360*cf5a6c84SAndroid Build Coastguard Worker break; // Unreachable code.
2361*cf5a6c84SAndroid Build Coastguard Worker }
2362*cf5a6c84SAndroid Build Coastguard Worker }
2363*cf5a6c84SAndroid Build Coastguard Worker
2364*cf5a6c84SAndroid Build Coastguard Worker if (!request.msg.rtm_family) request.msg.rtm_family = AF_INET;
2365*cf5a6c84SAndroid Build Coastguard Worker if (!tflag && opt == RTM_NEWRULE) request.msg.rtm_table = RT_TABLE_MAIN;
2366*cf5a6c84SAndroid Build Coastguard Worker
2367*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(0, 0, 0, &request, sizeof(request));
2368*cf5a6c84SAndroid Build Coastguard Worker return (filter_nlmesg(NULL, NULL));
2369*cf5a6c84SAndroid Build Coastguard Worker }
2370*cf5a6c84SAndroid Build Coastguard Worker
show_rules(struct nlmsghdr * mhdr,char ** argv)2371*cf5a6c84SAndroid Build Coastguard Worker static int show_rules(struct nlmsghdr *mhdr,
2372*cf5a6c84SAndroid Build Coastguard Worker char **argv __attribute__ ((__unused__)))
2373*cf5a6c84SAndroid Build Coastguard Worker {
2374*cf5a6c84SAndroid Build Coastguard Worker struct rtmsg *msg = NLMSG_DATA(mhdr);
2375*cf5a6c84SAndroid Build Coastguard Worker struct rtattr *rta, *attr[RTA_MAX+1] = {0,};
2376*cf5a6c84SAndroid Build Coastguard Worker int32_t tvar, msglen = mhdr->nlmsg_len - NLMSG_LENGTH(sizeof(struct rtmsg));
2377*cf5a6c84SAndroid Build Coastguard Worker int hlen = ((msg->rtm_family == AF_INET) ? 32
2378*cf5a6c84SAndroid Build Coastguard Worker : ((msg->rtm_family == AF_INET6) ? 128 : -1));
2379*cf5a6c84SAndroid Build Coastguard Worker
2380*cf5a6c84SAndroid Build Coastguard Worker if (mhdr->nlmsg_type != RTM_NEWRULE) return 0;
2381*cf5a6c84SAndroid Build Coastguard Worker if (msglen < 0) return 1;
2382*cf5a6c84SAndroid Build Coastguard Worker
2383*cf5a6c84SAndroid Build Coastguard Worker tvar = msglen;
2384*cf5a6c84SAndroid Build Coastguard Worker for (rta = RTM_RTA(msg); RTA_OK(rta, tvar); rta=RTA_NEXT(rta, tvar))
2385*cf5a6c84SAndroid Build Coastguard Worker if (rta->rta_type <= RTA_MAX) attr[rta->rta_type] = rta;
2386*cf5a6c84SAndroid Build Coastguard Worker
2387*cf5a6c84SAndroid Build Coastguard Worker if (tvar) error_msg("deficit %d, rtalen = %d!", tvar, rta->rta_len);
2388*cf5a6c84SAndroid Build Coastguard Worker
2389*cf5a6c84SAndroid Build Coastguard Worker printf("%u:\tfrom ", attr[RTA_PRIORITY] ?
2390*cf5a6c84SAndroid Build Coastguard Worker *(unsigned *)RTA_DATA(attr[RTA_PRIORITY]) : 0);
2391*cf5a6c84SAndroid Build Coastguard Worker
2392*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_SRC]) {
2393*cf5a6c84SAndroid Build Coastguard Worker printf("%s", (msg->rtm_family == AF_INET || msg->rtm_family == AF_INET6)
2394*cf5a6c84SAndroid Build Coastguard Worker ? inet_ntop(msg->rtm_family, RTA_DATA(attr[RTA_SRC]),
2395*cf5a6c84SAndroid Build Coastguard Worker toybuf, sizeof(toybuf))
2396*cf5a6c84SAndroid Build Coastguard Worker : "???");
2397*cf5a6c84SAndroid Build Coastguard Worker (msg->rtm_src_len != hlen) ? printf("/%u", msg->rtm_src_len) : 0;
2398*cf5a6c84SAndroid Build Coastguard Worker } else msg->rtm_src_len ? printf("0/%d", msg->rtm_src_len) : printf("all");
2399*cf5a6c84SAndroid Build Coastguard Worker
2400*cf5a6c84SAndroid Build Coastguard Worker xputc(' ');
2401*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_DST]) {
2402*cf5a6c84SAndroid Build Coastguard Worker printf("to %s", (msg->rtm_family == AF_INET || msg->rtm_family == AF_INET6)
2403*cf5a6c84SAndroid Build Coastguard Worker ? inet_ntop(msg->rtm_family, RTA_DATA(attr[RTA_DST]),
2404*cf5a6c84SAndroid Build Coastguard Worker toybuf, sizeof(toybuf)) : "???");
2405*cf5a6c84SAndroid Build Coastguard Worker (msg->rtm_dst_len != hlen) ? printf("/%u", msg->rtm_dst_len) : xputc(' ');
2406*cf5a6c84SAndroid Build Coastguard Worker } else if (msg->rtm_dst_len)
2407*cf5a6c84SAndroid Build Coastguard Worker printf("to 0/%d ", msg->rtm_dst_len);
2408*cf5a6c84SAndroid Build Coastguard Worker
2409*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_tos)
2410*cf5a6c84SAndroid Build Coastguard Worker printf("tos %s ", namefromRPDB(msg->rtm_tos, RPDB_rtdsfield));
2411*cf5a6c84SAndroid Build Coastguard Worker
2412*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_PROTOINFO])
2413*cf5a6c84SAndroid Build Coastguard Worker printf("fwmark %#x ", *(uint32_t*)RTA_DATA(attr[RTA_PROTOINFO]));
2414*cf5a6c84SAndroid Build Coastguard Worker
2415*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_IIF]) printf("iif %s ", (char*)RTA_DATA(attr[RTA_IIF]));
2416*cf5a6c84SAndroid Build Coastguard Worker
2417*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_table)
2418*cf5a6c84SAndroid Build Coastguard Worker printf("lookup %s ", namefromRPDB(msg->rtm_table, RPDB_rttables));
2419*cf5a6c84SAndroid Build Coastguard Worker
2420*cf5a6c84SAndroid Build Coastguard Worker if (attr[RTA_FLOW]) {
2421*cf5a6c84SAndroid Build Coastguard Worker u_int32_t from, to = *(u_int32_t *)RTA_DATA(attr[RTA_FLOW]);
2422*cf5a6c84SAndroid Build Coastguard Worker char *format = "realms %s/";
2423*cf5a6c84SAndroid Build Coastguard Worker
2424*cf5a6c84SAndroid Build Coastguard Worker to = (from = (to >> 16)) & 0xFFFF;
2425*cf5a6c84SAndroid Build Coastguard Worker format = (from ? format: "%s");
2426*cf5a6c84SAndroid Build Coastguard Worker printf(format, namefromRPDB((from ? from : to), RPDB_rtrealms));
2427*cf5a6c84SAndroid Build Coastguard Worker }
2428*cf5a6c84SAndroid Build Coastguard Worker
2429*cf5a6c84SAndroid Build Coastguard Worker if (msg->rtm_type == RTN_NAT) {
2430*cf5a6c84SAndroid Build Coastguard Worker if (!attr[RTA_GATEWAY]) printf("masquerade");
2431*cf5a6c84SAndroid Build Coastguard Worker else printf("map-to %s ", inet_ntop(msg->rtm_family,
2432*cf5a6c84SAndroid Build Coastguard Worker RTA_DATA(attr[RTA_GATEWAY]), toybuf, sizeof(toybuf)));
2433*cf5a6c84SAndroid Build Coastguard Worker } else if (msg->rtm_type != RTN_UNICAST)
2434*cf5a6c84SAndroid Build Coastguard Worker printf("%s", rtmtype_idx2str(msg->rtm_type));
2435*cf5a6c84SAndroid Build Coastguard Worker
2436*cf5a6c84SAndroid Build Coastguard Worker xputc('\n');
2437*cf5a6c84SAndroid Build Coastguard Worker return 0;
2438*cf5a6c84SAndroid Build Coastguard Worker }
2439*cf5a6c84SAndroid Build Coastguard Worker
rulelist(char ** argv)2440*cf5a6c84SAndroid Build Coastguard Worker static int rulelist(char **argv)
2441*cf5a6c84SAndroid Build Coastguard Worker {
2442*cf5a6c84SAndroid Build Coastguard Worker if (*argv) {
2443*cf5a6c84SAndroid Build Coastguard Worker error_msg("'ip rule show' does not take any arguments.");
2444*cf5a6c84SAndroid Build Coastguard Worker return 1;
2445*cf5a6c84SAndroid Build Coastguard Worker }
2446*cf5a6c84SAndroid Build Coastguard Worker send_nlmesg(RTM_GETRULE, NLM_F_ROOT | NLM_F_MATCH | NLM_F_REQUEST,
2447*cf5a6c84SAndroid Build Coastguard Worker ((TT.addressfamily != AF_UNSPEC) ? TT.addressfamily : AF_INET), NULL, 0);
2448*cf5a6c84SAndroid Build Coastguard Worker return filter_nlmesg(show_rules, argv);
2449*cf5a6c84SAndroid Build Coastguard Worker }
2450*cf5a6c84SAndroid Build Coastguard Worker
iprule(char ** argv)2451*cf5a6c84SAndroid Build Coastguard Worker static int iprule(char **argv)
2452*cf5a6c84SAndroid Build Coastguard Worker {
2453*cf5a6c84SAndroid Build Coastguard Worker int idx;
2454*cf5a6c84SAndroid Build Coastguard Worker struct arglist options[] = {{"add", 0}, {"delete", 0}, {"list", 1},
2455*cf5a6c84SAndroid Build Coastguard Worker {"show", 1}, {NULL, -1}};
2456*cf5a6c84SAndroid Build Coastguard Worker cmdobj ipcmd, cmdobjlist[] = {ruleupdate, rulelist};
2457*cf5a6c84SAndroid Build Coastguard Worker
2458*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) idx = 1;
2459*cf5a6c84SAndroid Build Coastguard Worker else if ((idx = substring_to_idx(*argv++, options)) == -1)
2460*cf5a6c84SAndroid Build Coastguard Worker show_iprule_help();
2461*cf5a6c84SAndroid Build Coastguard Worker ipcmd = cmdobjlist[idx];
2462*cf5a6c84SAndroid Build Coastguard Worker return ipcmd(argv);
2463*cf5a6c84SAndroid Build Coastguard Worker }
2464*cf5a6c84SAndroid Build Coastguard Worker //============================================================================
2465*cf5a6c84SAndroid Build Coastguard Worker // code for ip tunnel.
2466*cf5a6c84SAndroid Build Coastguard Worker //============================================================================
show_iptunnel_help(void)2467*cf5a6c84SAndroid Build Coastguard Worker static void show_iptunnel_help(void)
2468*cf5a6c84SAndroid Build Coastguard Worker {
2469*cf5a6c84SAndroid Build Coastguard Worker error_exit("usage: iptunnel { add | change | del | show } [NAME]\n"
2470*cf5a6c84SAndroid Build Coastguard Worker " [mode { ipip | gre | sit }] [remote ADDR] [local ADDR]\n"
2471*cf5a6c84SAndroid Build Coastguard Worker " [[i|o]seq] [[i|o]key KEY] [[i|o]csum] [ttl TTL]\n"
2472*cf5a6c84SAndroid Build Coastguard Worker " [tos TOS] [[no]pmtudisc] [dev PHYS_DEV]");
2473*cf5a6c84SAndroid Build Coastguard Worker }
2474*cf5a6c84SAndroid Build Coastguard Worker
tnl_ioctl(char * dev,int rtype,struct ip_tunnel_parm * ptnl)2475*cf5a6c84SAndroid Build Coastguard Worker static int tnl_ioctl(char *dev, int rtype, struct ip_tunnel_parm *ptnl)
2476*cf5a6c84SAndroid Build Coastguard Worker {
2477*cf5a6c84SAndroid Build Coastguard Worker struct ifreq req;
2478*cf5a6c84SAndroid Build Coastguard Worker int fd, ret = 0;
2479*cf5a6c84SAndroid Build Coastguard Worker
2480*cf5a6c84SAndroid Build Coastguard Worker if ((rtype == SIOCCHGTUNNEL || rtype == SIOCDELTUNNEL) && *ptnl->name)
2481*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(req.ifr_name, ptnl->name, IF_NAMESIZE);
2482*cf5a6c84SAndroid Build Coastguard Worker else xstrncpy(req.ifr_name, dev, IF_NAMESIZE);
2483*cf5a6c84SAndroid Build Coastguard Worker
2484*cf5a6c84SAndroid Build Coastguard Worker if (rtype != SIOCGIFHWADDR) req.ifr_ifru.ifru_data = (void*)ptnl;
2485*cf5a6c84SAndroid Build Coastguard Worker fd = xsocket(AF_INET, SOCK_DGRAM, 0);
2486*cf5a6c84SAndroid Build Coastguard Worker
2487*cf5a6c84SAndroid Build Coastguard Worker if (rtype == SIOCGETTUNNEL) ret = ioctl(fd, rtype, &req);
2488*cf5a6c84SAndroid Build Coastguard Worker else if (rtype == SIOCGIFHWADDR)
2489*cf5a6c84SAndroid Build Coastguard Worker ret = (ioctl(fd, rtype, &req) < 0) ? -1 : req.ifr_addr.sa_family;
2490*cf5a6c84SAndroid Build Coastguard Worker else xioctl(fd, rtype, &req);
2491*cf5a6c84SAndroid Build Coastguard Worker
2492*cf5a6c84SAndroid Build Coastguard Worker close(fd);
2493*cf5a6c84SAndroid Build Coastguard Worker return ret;
2494*cf5a6c84SAndroid Build Coastguard Worker }
2495*cf5a6c84SAndroid Build Coastguard Worker
display_tunnel(struct ip_tunnel_parm * ptnl)2496*cf5a6c84SAndroid Build Coastguard Worker static int display_tunnel(struct ip_tunnel_parm *ptnl)
2497*cf5a6c84SAndroid Build Coastguard Worker {
2498*cf5a6c84SAndroid Build Coastguard Worker char rmt_addr[64], lcl_addr[64], ikey_str[64], okey_str[64];
2499*cf5a6c84SAndroid Build Coastguard Worker
2500*cf5a6c84SAndroid Build Coastguard Worker printf("%s: %s/ip", ptnl->name, ptnl->iph.protocol == IPPROTO_IPIP ? "ip" :
2501*cf5a6c84SAndroid Build Coastguard Worker (ptnl->iph.protocol == IPPROTO_GRE ? "gre" :
2502*cf5a6c84SAndroid Build Coastguard Worker (ptnl->iph.protocol == IPPROTO_IPV6 ? "ipv6" : "unknown")));
2503*cf5a6c84SAndroid Build Coastguard Worker printf(" remote %s local %s ", ptnl->iph.daddr ?
2504*cf5a6c84SAndroid Build Coastguard Worker inet_ntop(AF_INET, &ptnl->iph.daddr, rmt_addr, sizeof(rmt_addr)) : "any",
2505*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.saddr ? inet_ntop(AF_INET, &ptnl->iph.saddr, lcl_addr,
2506*cf5a6c84SAndroid Build Coastguard Worker sizeof(lcl_addr)) : "any");
2507*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->link) {
2508*cf5a6c84SAndroid Build Coastguard Worker struct ifreq req;
2509*cf5a6c84SAndroid Build Coastguard Worker int fd;
2510*cf5a6c84SAndroid Build Coastguard Worker
2511*cf5a6c84SAndroid Build Coastguard Worker req.ifr_ifindex = ptnl->link;
2512*cf5a6c84SAndroid Build Coastguard Worker fd = xsocket(AF_INET, SOCK_DGRAM, 0);
2513*cf5a6c84SAndroid Build Coastguard Worker if (ioctl(fd, SIOCGIFNAME, &req) < 0) perror_msg("SIOCGIFNAME");
2514*cf5a6c84SAndroid Build Coastguard Worker else printf(" dev %s ", req.ifr_name);
2515*cf5a6c84SAndroid Build Coastguard Worker close(fd);
2516*cf5a6c84SAndroid Build Coastguard Worker }
2517*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->iph.ttl) printf(" ttl %d ", ptnl->iph.ttl);
2518*cf5a6c84SAndroid Build Coastguard Worker else printf(" ttl inherit ");
2519*cf5a6c84SAndroid Build Coastguard Worker
2520*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->iph.tos) {
2521*cf5a6c84SAndroid Build Coastguard Worker printf(" tos");
2522*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->iph.tos & 1) printf(" inherit");
2523*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->iph.tos & ~1) printf("%c%s ", ptnl->iph.tos & 1 ? '/' : ' ',
2524*cf5a6c84SAndroid Build Coastguard Worker namefromRPDB((ptnl->iph.tos & ~1), RPDB_rtdsfield));
2525*cf5a6c84SAndroid Build Coastguard Worker }
2526*cf5a6c84SAndroid Build Coastguard Worker if (!(ptnl->iph.frag_off & htons(IP_DF))) printf(" nopmtudisc");
2527*cf5a6c84SAndroid Build Coastguard Worker inet_ntop(AF_INET, &ptnl->i_key, ikey_str, sizeof(ikey_str));
2528*cf5a6c84SAndroid Build Coastguard Worker if ((ptnl->i_flags & GRE_KEY) && (ptnl->o_flags & GRE_KEY)
2529*cf5a6c84SAndroid Build Coastguard Worker && ptnl->o_key == ptnl->i_key) printf(" key %s", ikey_str);
2530*cf5a6c84SAndroid Build Coastguard Worker else if ((ptnl->i_flags | ptnl->o_flags) & GRE_KEY) {
2531*cf5a6c84SAndroid Build Coastguard Worker inet_ntop(AF_INET, &ptnl->o_key, okey_str, sizeof(okey_str));
2532*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->i_flags & GRE_KEY) printf(" ikey %s ", ikey_str);
2533*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->o_flags & GRE_KEY) printf(" okey %s ", okey_str);
2534*cf5a6c84SAndroid Build Coastguard Worker }
2535*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->i_flags & GRE_SEQ) printf("\n Drop packets out of sequence.\n");
2536*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->i_flags & GRE_CSUM)
2537*cf5a6c84SAndroid Build Coastguard Worker printf("\n Checksum in received packet is required.");
2538*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->o_flags & GRE_SEQ) printf("\n Sequence packets on output.");
2539*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->o_flags & GRE_CSUM) printf("\n Checksum output packets.");
2540*cf5a6c84SAndroid Build Coastguard Worker xputc('\n');
2541*cf5a6c84SAndroid Build Coastguard Worker return 0;
2542*cf5a6c84SAndroid Build Coastguard Worker }
2543*cf5a6c84SAndroid Build Coastguard Worker
read_tunnel(struct ip_tunnel_parm * ptnl)2544*cf5a6c84SAndroid Build Coastguard Worker static int read_tunnel(struct ip_tunnel_parm *ptnl)
2545*cf5a6c84SAndroid Build Coastguard Worker {
2546*cf5a6c84SAndroid Build Coastguard Worker int count = 0;
2547*cf5a6c84SAndroid Build Coastguard Worker char iface[IF_NAMESIZE];
2548*cf5a6c84SAndroid Build Coastguard Worker struct ip_tunnel_parm iptnl;
2549*cf5a6c84SAndroid Build Coastguard Worker FILE *fp = xfopen("/proc/net/dev", "r");
2550*cf5a6c84SAndroid Build Coastguard Worker
2551*cf5a6c84SAndroid Build Coastguard Worker while (fgets(toybuf, sizeof(toybuf), fp)) {
2552*cf5a6c84SAndroid Build Coastguard Worker char *ptr;
2553*cf5a6c84SAndroid Build Coastguard Worker int ret;
2554*cf5a6c84SAndroid Build Coastguard Worker
2555*cf5a6c84SAndroid Build Coastguard Worker if (count++ < 2) continue; // 1st two lines are header.
2556*cf5a6c84SAndroid Build Coastguard Worker
2557*cf5a6c84SAndroid Build Coastguard Worker ptr = strchr(toybuf, ':');
2558*cf5a6c84SAndroid Build Coastguard Worker if (!ptr || (*ptr++ = 0, sscanf(toybuf, "%s", iface) != 1))
2559*cf5a6c84SAndroid Build Coastguard Worker error_exit("invalid format of '/proc/net/dev'");
2560*cf5a6c84SAndroid Build Coastguard Worker if (*ptnl->name && strcmp(ptnl->name, iface)) continue;
2561*cf5a6c84SAndroid Build Coastguard Worker if ((ret = tnl_ioctl(iface, SIOCGIFHWADDR, &iptnl)) < 0) {
2562*cf5a6c84SAndroid Build Coastguard Worker error_msg("failed to get type of '%s'", iface);
2563*cf5a6c84SAndroid Build Coastguard Worker continue;
2564*cf5a6c84SAndroid Build Coastguard Worker }
2565*cf5a6c84SAndroid Build Coastguard Worker if (ret != ARPHRD_TUNNEL && ret != ARPHRD_SIT &&
2566*cf5a6c84SAndroid Build Coastguard Worker ret != ARPHRD_IPGRE) continue;
2567*cf5a6c84SAndroid Build Coastguard Worker
2568*cf5a6c84SAndroid Build Coastguard Worker memset(&iptnl, 0, sizeof(iptnl));
2569*cf5a6c84SAndroid Build Coastguard Worker if (tnl_ioctl(iface, SIOCGETTUNNEL, &iptnl) < 0) continue;
2570*cf5a6c84SAndroid Build Coastguard Worker if ((ptnl->link && iptnl.link != ptnl->link) || (*ptnl->name &&
2571*cf5a6c84SAndroid Build Coastguard Worker strcmp(iptnl.name, ptnl->name)) || (ptnl->iph.daddr &&
2572*cf5a6c84SAndroid Build Coastguard Worker iptnl.iph.daddr != ptnl->iph.daddr) || (ptnl->iph.saddr &&
2573*cf5a6c84SAndroid Build Coastguard Worker iptnl.iph.saddr != ptnl->iph.saddr) || (ptnl->i_key &&
2574*cf5a6c84SAndroid Build Coastguard Worker iptnl.i_key != ptnl->i_key)) continue;
2575*cf5a6c84SAndroid Build Coastguard Worker display_tunnel(&iptnl);
2576*cf5a6c84SAndroid Build Coastguard Worker }
2577*cf5a6c84SAndroid Build Coastguard Worker fclose(fp);
2578*cf5a6c84SAndroid Build Coastguard Worker return 0;
2579*cf5a6c84SAndroid Build Coastguard Worker }
2580*cf5a6c84SAndroid Build Coastguard Worker
parse_iptunnel_args(struct ip_tunnel_parm * ptnl,char ** argv,int ipt_opt_idx)2581*cf5a6c84SAndroid Build Coastguard Worker static void parse_iptunnel_args(struct ip_tunnel_parm *ptnl, char **argv,
2582*cf5a6c84SAndroid Build Coastguard Worker int ipt_opt_idx)
2583*cf5a6c84SAndroid Build Coastguard Worker {
2584*cf5a6c84SAndroid Build Coastguard Worker int idx;
2585*cf5a6c84SAndroid Build Coastguard Worker uint8_t af = AF_INET;
2586*cf5a6c84SAndroid Build Coastguard Worker uint32_t addr = 0;
2587*cf5a6c84SAndroid Build Coastguard Worker struct arglist opts[] = { {"mode", 0}, {"key", 1}, {"ikey", 2},
2588*cf5a6c84SAndroid Build Coastguard Worker {"okey", 3}, {"seq", 4}, {"iseq", 5}, {"oseq", 6}, {"csum", 7},
2589*cf5a6c84SAndroid Build Coastguard Worker {"icsum", 8}, {"ocsum", 9}, {"nopmtudisc", 10}, {"pmtudisc", 11},
2590*cf5a6c84SAndroid Build Coastguard Worker {"remote", 12}, {"local", 13},{"dev", 14}, {"ttl", 15}, {"tos", 16},
2591*cf5a6c84SAndroid Build Coastguard Worker {"dsfield", 17}, {"name", 18}, {NULL, -1}
2592*cf5a6c84SAndroid Build Coastguard Worker };
2593*cf5a6c84SAndroid Build Coastguard Worker
2594*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.version = 4; // The value indicates the version of IP (4 or 6)
2595*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.ihl = 5; // Minimum Internet Header Length
2596*cf5a6c84SAndroid Build Coastguard Worker // frag_off is measured in units of 8 octets (64 bits)
2597*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.frag_off = htons(IP_DF);
2598*cf5a6c84SAndroid Build Coastguard Worker if (*argv && ipt_opt_idx <= 2 && string_to_idx(*argv, opts) == -1) {
2599*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(ptnl->name, *argv, IF_NAMESIZE);
2600*cf5a6c84SAndroid Build Coastguard Worker if (ipt_opt_idx == 1) {
2601*cf5a6c84SAndroid Build Coastguard Worker struct ip_tunnel_parm iptnl_old;
2602*cf5a6c84SAndroid Build Coastguard Worker
2603*cf5a6c84SAndroid Build Coastguard Worker memset(&iptnl_old, 0, sizeof(iptnl_old));
2604*cf5a6c84SAndroid Build Coastguard Worker tnl_ioctl(ptnl->name, SIOCGETTUNNEL, &iptnl_old);
2605*cf5a6c84SAndroid Build Coastguard Worker *ptnl = iptnl_old;
2606*cf5a6c84SAndroid Build Coastguard Worker }
2607*cf5a6c84SAndroid Build Coastguard Worker argv++;
2608*cf5a6c84SAndroid Build Coastguard Worker }
2609*cf5a6c84SAndroid Build Coastguard Worker for (; *argv; argv++, addr = 0) {
2610*cf5a6c84SAndroid Build Coastguard Worker switch (idx = string_to_idx(*argv, opts)) {
2611*cf5a6c84SAndroid Build Coastguard Worker case 0:
2612*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("mode is missing");
2613*cf5a6c84SAndroid Build Coastguard Worker if ((!strcmp("ipip", *argv) || !strcmp("ip/ip", *argv)))
2614*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.protocol = IPPROTO_IPIP;
2615*cf5a6c84SAndroid Build Coastguard Worker else if ((!strcmp("gre", *argv) || !strcmp("gre/ip", *argv)))
2616*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.protocol = IPPROTO_GRE;
2617*cf5a6c84SAndroid Build Coastguard Worker else if ((!strcmp("sit", *argv) || !strcmp("ipv6/ip", *argv)))
2618*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.protocol = IPPROTO_IPV6;
2619*cf5a6c84SAndroid Build Coastguard Worker else show_iptunnel_help();
2620*cf5a6c84SAndroid Build Coastguard Worker break;
2621*cf5a6c84SAndroid Build Coastguard Worker case 1:
2622*cf5a6c84SAndroid Build Coastguard Worker case 2:
2623*cf5a6c84SAndroid Build Coastguard Worker case 3:
2624*cf5a6c84SAndroid Build Coastguard Worker {
2625*cf5a6c84SAndroid Build Coastguard Worker struct addrinfo *info, hint;
2626*cf5a6c84SAndroid Build Coastguard Worker int ret;
2627*cf5a6c84SAndroid Build Coastguard Worker
2628*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("key value is missing");
2629*cf5a6c84SAndroid Build Coastguard Worker memset(&hint, 0, sizeof(hint));
2630*cf5a6c84SAndroid Build Coastguard Worker hint.ai_family = AF_INET;
2631*cf5a6c84SAndroid Build Coastguard Worker ret = getaddrinfo(*argv, NULL, &hint, &info);
2632*cf5a6c84SAndroid Build Coastguard Worker if (ret || !info) error_exit("invalid argument to key");
2633*cf5a6c84SAndroid Build Coastguard Worker freeaddrinfo(info);
2634*cf5a6c84SAndroid Build Coastguard Worker
2635*cf5a6c84SAndroid Build Coastguard Worker if (strchr(*argv, '.')) {
2636*cf5a6c84SAndroid Build Coastguard Worker if (get_prefix(&addr, &af, *argv, AF_INET))
2637*cf5a6c84SAndroid Build Coastguard Worker error_exit("invalid key '%s'", *argv);
2638*cf5a6c84SAndroid Build Coastguard Worker } else {
2639*cf5a6c84SAndroid Build Coastguard Worker unsigned key_val;
2640*cf5a6c84SAndroid Build Coastguard Worker
2641*cf5a6c84SAndroid Build Coastguard Worker sscanf(*argv, "%u", &key_val);
2642*cf5a6c84SAndroid Build Coastguard Worker addr = htonl(key_val);
2643*cf5a6c84SAndroid Build Coastguard Worker }
2644*cf5a6c84SAndroid Build Coastguard Worker if (idx == 1) {
2645*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_flags |= GRE_KEY;
2646*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_flags |= GRE_KEY;
2647*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_key = ptnl->o_key = addr;
2648*cf5a6c84SAndroid Build Coastguard Worker } else if (idx == 2) {
2649*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_flags |= GRE_KEY;
2650*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_key = addr;
2651*cf5a6c84SAndroid Build Coastguard Worker } else {
2652*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_flags |= GRE_KEY;
2653*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_key = addr;
2654*cf5a6c84SAndroid Build Coastguard Worker }
2655*cf5a6c84SAndroid Build Coastguard Worker }
2656*cf5a6c84SAndroid Build Coastguard Worker break;
2657*cf5a6c84SAndroid Build Coastguard Worker case 4:
2658*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_flags |= GRE_SEQ;
2659*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_flags |= GRE_SEQ;
2660*cf5a6c84SAndroid Build Coastguard Worker break;
2661*cf5a6c84SAndroid Build Coastguard Worker case 5:
2662*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_flags |= GRE_SEQ;
2663*cf5a6c84SAndroid Build Coastguard Worker break;
2664*cf5a6c84SAndroid Build Coastguard Worker case 6:
2665*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_flags |= GRE_SEQ;
2666*cf5a6c84SAndroid Build Coastguard Worker break;
2667*cf5a6c84SAndroid Build Coastguard Worker case 7:
2668*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_flags |= GRE_CSUM;
2669*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_flags |= GRE_CSUM;
2670*cf5a6c84SAndroid Build Coastguard Worker break;
2671*cf5a6c84SAndroid Build Coastguard Worker case 8:
2672*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_flags |= GRE_CSUM;
2673*cf5a6c84SAndroid Build Coastguard Worker break;
2674*cf5a6c84SAndroid Build Coastguard Worker case 9:
2675*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_flags |= GRE_CSUM;
2676*cf5a6c84SAndroid Build Coastguard Worker break;
2677*cf5a6c84SAndroid Build Coastguard Worker case 10:
2678*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.frag_off = 0;
2679*cf5a6c84SAndroid Build Coastguard Worker break;
2680*cf5a6c84SAndroid Build Coastguard Worker case 11:
2681*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.frag_off = htons(IP_DF);
2682*cf5a6c84SAndroid Build Coastguard Worker break;
2683*cf5a6c84SAndroid Build Coastguard Worker case 12:
2684*cf5a6c84SAndroid Build Coastguard Worker case 13:
2685*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("remote/local address is missing");
2686*cf5a6c84SAndroid Build Coastguard Worker if (get_prefix(&addr, &af, *argv, AF_INET))
2687*cf5a6c84SAndroid Build Coastguard Worker error_exit("invalid remote/local address '%s'", *argv);
2688*cf5a6c84SAndroid Build Coastguard Worker (idx == 12) ? (ptnl->iph.daddr = addr) : (ptnl->iph.saddr = addr);
2689*cf5a6c84SAndroid Build Coastguard Worker break;
2690*cf5a6c84SAndroid Build Coastguard Worker case 14:
2691*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("device name is missing");
2692*cf5a6c84SAndroid Build Coastguard Worker else {
2693*cf5a6c84SAndroid Build Coastguard Worker struct ifreq req;
2694*cf5a6c84SAndroid Build Coastguard Worker int fd;
2695*cf5a6c84SAndroid Build Coastguard Worker
2696*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(req.ifr_name, *argv, IFNAMSIZ);
2697*cf5a6c84SAndroid Build Coastguard Worker fd = xsocket(AF_INET, SOCK_DGRAM, 0);
2698*cf5a6c84SAndroid Build Coastguard Worker xioctl(fd, SIOCGIFINDEX, &req);
2699*cf5a6c84SAndroid Build Coastguard Worker close(fd);
2700*cf5a6c84SAndroid Build Coastguard Worker ptnl->link = req.ifr_ifindex;
2701*cf5a6c84SAndroid Build Coastguard Worker }
2702*cf5a6c84SAndroid Build Coastguard Worker break;
2703*cf5a6c84SAndroid Build Coastguard Worker case 15:
2704*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("ttl value is missing");
2705*cf5a6c84SAndroid Build Coastguard Worker if (strcmp(*argv, "inherit"))
2706*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.ttl = atolx_range(*argv, 0, 255);
2707*cf5a6c84SAndroid Build Coastguard Worker break;
2708*cf5a6c84SAndroid Build Coastguard Worker case 16:
2709*cf5a6c84SAndroid Build Coastguard Worker case 17:
2710*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("tos value is missing");
2711*cf5a6c84SAndroid Build Coastguard Worker if (strcmp(*argv, "inherit")) {
2712*cf5a6c84SAndroid Build Coastguard Worker char *ptr;
2713*cf5a6c84SAndroid Build Coastguard Worker unsigned long tval = strtoul(*argv, &ptr, 16);
2714*cf5a6c84SAndroid Build Coastguard Worker
2715*cf5a6c84SAndroid Build Coastguard Worker if (tval > 255) error_exit("invalid tos value '%s'", *argv);
2716*cf5a6c84SAndroid Build Coastguard Worker if (*ptr) {
2717*cf5a6c84SAndroid Build Coastguard Worker int ret;
2718*cf5a6c84SAndroid Build Coastguard Worker
2719*cf5a6c84SAndroid Build Coastguard Worker if ((ret = idxfromRPDB(*argv, RPDB_rtdsfield)) < 0)
2720*cf5a6c84SAndroid Build Coastguard Worker error_exit("invalid tos value");
2721*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.tos = ret;
2722*cf5a6c84SAndroid Build Coastguard Worker } else ptnl->iph.tos = tval;
2723*cf5a6c84SAndroid Build Coastguard Worker } else ptnl->iph.tos = 1;
2724*cf5a6c84SAndroid Build Coastguard Worker break;
2725*cf5a6c84SAndroid Build Coastguard Worker case 18:
2726*cf5a6c84SAndroid Build Coastguard Worker if (*ptnl->name) error_exit("invalid tunnel");
2727*cf5a6c84SAndroid Build Coastguard Worker else {
2728*cf5a6c84SAndroid Build Coastguard Worker if (!*++argv) error_exit("name is missing");
2729*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(ptnl->name, *argv, IF_NAMESIZE);
2730*cf5a6c84SAndroid Build Coastguard Worker }
2731*cf5a6c84SAndroid Build Coastguard Worker break;
2732*cf5a6c84SAndroid Build Coastguard Worker default:
2733*cf5a6c84SAndroid Build Coastguard Worker if (*ptnl->name) error_exit("invalid tunnel");
2734*cf5a6c84SAndroid Build Coastguard Worker xstrncpy(ptnl->name, *argv, IF_NAMESIZE);
2735*cf5a6c84SAndroid Build Coastguard Worker break;
2736*cf5a6c84SAndroid Build Coastguard Worker }
2737*cf5a6c84SAndroid Build Coastguard Worker }
2738*cf5a6c84SAndroid Build Coastguard Worker if (ptnl->iph.protocol == IPPROTO_IPIP ||
2739*cf5a6c84SAndroid Build Coastguard Worker ptnl->iph.protocol == IPPROTO_IPV6) {
2740*cf5a6c84SAndroid Build Coastguard Worker if ((ptnl->i_flags & GRE_KEY) || (ptnl->o_flags & GRE_KEY))
2741*cf5a6c84SAndroid Build Coastguard Worker error_exit("[i|o]key is allowed with gre only");
2742*cf5a6c84SAndroid Build Coastguard Worker if ((ptnl->i_flags & GRE_SEQ) || (ptnl->o_flags & GRE_SEQ))
2743*cf5a6c84SAndroid Build Coastguard Worker error_exit("[i|o]seq is allowed with gre only");
2744*cf5a6c84SAndroid Build Coastguard Worker if ((ptnl->i_flags & GRE_CSUM) || (ptnl->o_flags & GRE_CSUM))
2745*cf5a6c84SAndroid Build Coastguard Worker error_exit("[i|o]csum is allowed with gre only");
2746*cf5a6c84SAndroid Build Coastguard Worker }
2747*cf5a6c84SAndroid Build Coastguard Worker if (!ptnl->i_key && IN_MULTICAST(ntohl(ptnl->iph.daddr))) {
2748*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_key = ptnl->iph.daddr;
2749*cf5a6c84SAndroid Build Coastguard Worker ptnl->i_flags |= GRE_KEY;
2750*cf5a6c84SAndroid Build Coastguard Worker }
2751*cf5a6c84SAndroid Build Coastguard Worker if (!ptnl->o_key && IN_MULTICAST(ntohl(ptnl->iph.daddr))) {
2752*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_key = ptnl->iph.daddr;
2753*cf5a6c84SAndroid Build Coastguard Worker ptnl->o_flags |= GRE_KEY;
2754*cf5a6c84SAndroid Build Coastguard Worker }
2755*cf5a6c84SAndroid Build Coastguard Worker if (IN_MULTICAST(ntohl(ptnl->iph.daddr)) && !ptnl->iph.saddr)
2756*cf5a6c84SAndroid Build Coastguard Worker error_exit("broadcast tunnel requires a source address");
2757*cf5a6c84SAndroid Build Coastguard Worker }
2758*cf5a6c84SAndroid Build Coastguard Worker
tunnellist(char ** argv)2759*cf5a6c84SAndroid Build Coastguard Worker static int tunnellist(char **argv)
2760*cf5a6c84SAndroid Build Coastguard Worker {
2761*cf5a6c84SAndroid Build Coastguard Worker struct ip_tunnel_parm iptnl;
2762*cf5a6c84SAndroid Build Coastguard Worker int ret = 0;
2763*cf5a6c84SAndroid Build Coastguard Worker
2764*cf5a6c84SAndroid Build Coastguard Worker memset(&iptnl, 0, sizeof(iptnl));
2765*cf5a6c84SAndroid Build Coastguard Worker parse_iptunnel_args(&iptnl, argv, 3);
2766*cf5a6c84SAndroid Build Coastguard Worker
2767*cf5a6c84SAndroid Build Coastguard Worker if (iptnl.iph.protocol == IPPROTO_IPIP)
2768*cf5a6c84SAndroid Build Coastguard Worker ret = tnl_ioctl(*iptnl.name ? iptnl.name : "tunl0", SIOCGETTUNNEL, &iptnl);
2769*cf5a6c84SAndroid Build Coastguard Worker else if (iptnl.iph.protocol == IPPROTO_GRE)
2770*cf5a6c84SAndroid Build Coastguard Worker ret = tnl_ioctl(*iptnl.name ? iptnl.name : "gre0", SIOCGETTUNNEL, &iptnl);
2771*cf5a6c84SAndroid Build Coastguard Worker else if (iptnl.iph.protocol == IPPROTO_IPV6)
2772*cf5a6c84SAndroid Build Coastguard Worker ret = tnl_ioctl(*iptnl.name ? iptnl.name : "sit0", SIOCGETTUNNEL, &iptnl);
2773*cf5a6c84SAndroid Build Coastguard Worker else return read_tunnel(&iptnl);
2774*cf5a6c84SAndroid Build Coastguard Worker
2775*cf5a6c84SAndroid Build Coastguard Worker if (ret < 0) {
2776*cf5a6c84SAndroid Build Coastguard Worker perror_msg("SIOCGETTUNNEL");
2777*cf5a6c84SAndroid Build Coastguard Worker return ret;
2778*cf5a6c84SAndroid Build Coastguard Worker } else return display_tunnel(&iptnl);
2779*cf5a6c84SAndroid Build Coastguard Worker }
2780*cf5a6c84SAndroid Build Coastguard Worker
2781*cf5a6c84SAndroid Build Coastguard Worker // Performing add, change, & delete tunnel action, according to passed req_type
tunnelupdate(char ** argv)2782*cf5a6c84SAndroid Build Coastguard Worker static int tunnelupdate(char **argv)
2783*cf5a6c84SAndroid Build Coastguard Worker {
2784*cf5a6c84SAndroid Build Coastguard Worker struct ip_tunnel_parm iptnl;
2785*cf5a6c84SAndroid Build Coastguard Worker int idx = 2, rtype = SIOCDELTUNNEL;
2786*cf5a6c84SAndroid Build Coastguard Worker
2787*cf5a6c84SAndroid Build Coastguard Worker if (*argv[-1] == 'a') {
2788*cf5a6c84SAndroid Build Coastguard Worker idx = 0;
2789*cf5a6c84SAndroid Build Coastguard Worker rtype = SIOCADDTUNNEL;
2790*cf5a6c84SAndroid Build Coastguard Worker } else if (*argv[-1] == 'c') {
2791*cf5a6c84SAndroid Build Coastguard Worker idx = 1;
2792*cf5a6c84SAndroid Build Coastguard Worker rtype = SIOCCHGTUNNEL;
2793*cf5a6c84SAndroid Build Coastguard Worker }
2794*cf5a6c84SAndroid Build Coastguard Worker
2795*cf5a6c84SAndroid Build Coastguard Worker memset(&iptnl, 0, sizeof(iptnl));
2796*cf5a6c84SAndroid Build Coastguard Worker parse_iptunnel_args(&iptnl, argv, idx);
2797*cf5a6c84SAndroid Build Coastguard Worker if (idx != 2 && iptnl.iph.ttl && !(iptnl.iph.frag_off))
2798*cf5a6c84SAndroid Build Coastguard Worker error_exit("ttl > 0 and nopmtudisc are incompatible");
2799*cf5a6c84SAndroid Build Coastguard Worker if (iptnl.iph.protocol == IPPROTO_IPIP)
2800*cf5a6c84SAndroid Build Coastguard Worker return (tnl_ioctl("tunl0", rtype, &iptnl) < 0) ? 1 : 0;
2801*cf5a6c84SAndroid Build Coastguard Worker else if (iptnl.iph.protocol == IPPROTO_GRE)
2802*cf5a6c84SAndroid Build Coastguard Worker return (tnl_ioctl("gre0", rtype, &iptnl) < 0) ? 1 : 0;
2803*cf5a6c84SAndroid Build Coastguard Worker else if (iptnl.iph.protocol == IPPROTO_IPV6)
2804*cf5a6c84SAndroid Build Coastguard Worker return (tnl_ioctl("sit0", rtype, &iptnl) < 0) ? 1 : 0;
2805*cf5a6c84SAndroid Build Coastguard Worker else {
2806*cf5a6c84SAndroid Build Coastguard Worker if (idx != 2) error_exit("invalid tunnel mode");
2807*cf5a6c84SAndroid Build Coastguard Worker return (tnl_ioctl(iptnl.name, rtype, &iptnl) < 0) ? 1 : 0;
2808*cf5a6c84SAndroid Build Coastguard Worker }
2809*cf5a6c84SAndroid Build Coastguard Worker }
2810*cf5a6c84SAndroid Build Coastguard Worker
iptunnel(char ** argv)2811*cf5a6c84SAndroid Build Coastguard Worker static int iptunnel(char **argv)
2812*cf5a6c84SAndroid Build Coastguard Worker {
2813*cf5a6c84SAndroid Build Coastguard Worker int idx;
2814*cf5a6c84SAndroid Build Coastguard Worker struct arglist opts[] = {{"add", 0}, {"change", 0}, {"del", 0},
2815*cf5a6c84SAndroid Build Coastguard Worker {"delete", 0}, {"show", 1}, {"list", 1}, {"lst", 1}, {NULL, -1}
2816*cf5a6c84SAndroid Build Coastguard Worker };
2817*cf5a6c84SAndroid Build Coastguard Worker cmdobj ipcmd, cmdobjlist[] = {tunnelupdate, tunnellist};
2818*cf5a6c84SAndroid Build Coastguard Worker
2819*cf5a6c84SAndroid Build Coastguard Worker if (!*argv) idx = 1;
2820*cf5a6c84SAndroid Build Coastguard Worker else if ((idx = substring_to_idx(*argv++, opts)) == -1)
2821*cf5a6c84SAndroid Build Coastguard Worker show_iptunnel_help();
2822*cf5a6c84SAndroid Build Coastguard Worker ipcmd = cmdobjlist[idx];
2823*cf5a6c84SAndroid Build Coastguard Worker return ipcmd(argv);
2824*cf5a6c84SAndroid Build Coastguard Worker }
2825*cf5a6c84SAndroid Build Coastguard Worker
2826*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
2827*cf5a6c84SAndroid Build Coastguard Worker // Common code, which is used for all ip options.
2828*cf5a6c84SAndroid Build Coastguard Worker // ===========================================================================
2829*cf5a6c84SAndroid Build Coastguard Worker
2830*cf5a6c84SAndroid Build Coastguard Worker // Parse netlink messages and call input callback handler for action
filter_nlmesg(int (* fun)(struct nlmsghdr * mhdr,char ** argv),char ** argv)2831*cf5a6c84SAndroid Build Coastguard Worker static int filter_nlmesg(int (*fun)(struct nlmsghdr *mhdr, char **argv),
2832*cf5a6c84SAndroid Build Coastguard Worker char **argv)
2833*cf5a6c84SAndroid Build Coastguard Worker {
2834*cf5a6c84SAndroid Build Coastguard Worker while (1) {
2835*cf5a6c84SAndroid Build Coastguard Worker struct nlmsghdr *mhdr;
2836*cf5a6c84SAndroid Build Coastguard Worker int msglen = recv(TT.sockfd, TT.gbuf, MESG_LEN, 0);
2837*cf5a6c84SAndroid Build Coastguard Worker
2838*cf5a6c84SAndroid Build Coastguard Worker if ((msglen < 0) && (errno == EINTR || errno == EAGAIN)) continue;
2839*cf5a6c84SAndroid Build Coastguard Worker else if (msglen < 0) {
2840*cf5a6c84SAndroid Build Coastguard Worker error_msg("netlink receive error %s", strerror(errno));
2841*cf5a6c84SAndroid Build Coastguard Worker return 1;
2842*cf5a6c84SAndroid Build Coastguard Worker } else if (!msglen) {
2843*cf5a6c84SAndroid Build Coastguard Worker error_msg("EOF on netlink");
2844*cf5a6c84SAndroid Build Coastguard Worker return 1;
2845*cf5a6c84SAndroid Build Coastguard Worker }
2846*cf5a6c84SAndroid Build Coastguard Worker
2847*cf5a6c84SAndroid Build Coastguard Worker for (mhdr = (struct nlmsghdr*)TT.gbuf; NLMSG_OK(mhdr, msglen);
2848*cf5a6c84SAndroid Build Coastguard Worker mhdr = NLMSG_NEXT(mhdr, msglen)) {
2849*cf5a6c84SAndroid Build Coastguard Worker int err;
2850*cf5a6c84SAndroid Build Coastguard Worker if (mhdr->nlmsg_pid != getpid())
2851*cf5a6c84SAndroid Build Coastguard Worker continue;
2852*cf5a6c84SAndroid Build Coastguard Worker switch (mhdr->nlmsg_type) {
2853*cf5a6c84SAndroid Build Coastguard Worker case NLMSG_DONE:
2854*cf5a6c84SAndroid Build Coastguard Worker return 0;
2855*cf5a6c84SAndroid Build Coastguard Worker case NLMSG_ERROR:
2856*cf5a6c84SAndroid Build Coastguard Worker {
2857*cf5a6c84SAndroid Build Coastguard Worker struct nlmsgerr *merr = (struct nlmsgerr*)NLMSG_DATA(mhdr);
2858*cf5a6c84SAndroid Build Coastguard Worker
2859*cf5a6c84SAndroid Build Coastguard Worker if (merr->error == 0) return 0;
2860*cf5a6c84SAndroid Build Coastguard Worker if (mhdr->nlmsg_len < NLMSG_LENGTH(sizeof(struct nlmsgerr)))
2861*cf5a6c84SAndroid Build Coastguard Worker error_msg("ERROR truncated");
2862*cf5a6c84SAndroid Build Coastguard Worker else {
2863*cf5a6c84SAndroid Build Coastguard Worker errno = -merr->error;
2864*cf5a6c84SAndroid Build Coastguard Worker perror_msg("RTNETLINK answers");
2865*cf5a6c84SAndroid Build Coastguard Worker }
2866*cf5a6c84SAndroid Build Coastguard Worker return 1;
2867*cf5a6c84SAndroid Build Coastguard Worker }
2868*cf5a6c84SAndroid Build Coastguard Worker default:
2869*cf5a6c84SAndroid Build Coastguard Worker if (fun && (err = fun(mhdr, argv))) return err;
2870*cf5a6c84SAndroid Build Coastguard Worker break;
2871*cf5a6c84SAndroid Build Coastguard Worker }
2872*cf5a6c84SAndroid Build Coastguard Worker } // End of for loop.
2873*cf5a6c84SAndroid Build Coastguard Worker } // End of while loop.
2874*cf5a6c84SAndroid Build Coastguard Worker return 0;
2875*cf5a6c84SAndroid Build Coastguard Worker }
2876*cf5a6c84SAndroid Build Coastguard Worker
ip_main(void)2877*cf5a6c84SAndroid Build Coastguard Worker void ip_main(void)
2878*cf5a6c84SAndroid Build Coastguard Worker {
2879*cf5a6c84SAndroid Build Coastguard Worker char **optargv = toys.argv;
2880*cf5a6c84SAndroid Build Coastguard Worker int idx, isip = !(toys.which->name[2]); //1 -> if only ip
2881*cf5a6c84SAndroid Build Coastguard Worker cmdobj ipcmd, cmdobjlist[] = {ipaddr, iplink, iproute, iprule, iptunnel};
2882*cf5a6c84SAndroid Build Coastguard Worker
2883*cf5a6c84SAndroid Build Coastguard Worker for (++optargv; *optargv; ++optargv) {
2884*cf5a6c84SAndroid Build Coastguard Worker char *ptr = *optargv;
2885*cf5a6c84SAndroid Build Coastguard Worker struct arglist ip_options[] = {{"oneline", 0}, {"family", 1},
2886*cf5a6c84SAndroid Build Coastguard Worker {"4", 1}, {"6", 1}, {"0", 1}, {"stats", 2}, {NULL, -1}};
2887*cf5a6c84SAndroid Build Coastguard Worker
2888*cf5a6c84SAndroid Build Coastguard Worker if (*ptr != '-') break;
2889*cf5a6c84SAndroid Build Coastguard Worker else if ((*(ptr+1) == '-') && (*(ptr+2))) ptr +=2;
2890*cf5a6c84SAndroid Build Coastguard Worker //escape "--" and stop ip arg parsing.
2891*cf5a6c84SAndroid Build Coastguard Worker else if ((*(ptr+1) == '-') && (!*(ptr+2))) {
2892*cf5a6c84SAndroid Build Coastguard Worker *ptr +=1;
2893*cf5a6c84SAndroid Build Coastguard Worker break;
2894*cf5a6c84SAndroid Build Coastguard Worker } else ptr +=1;
2895*cf5a6c84SAndroid Build Coastguard Worker switch (substring_to_idx(ptr, ip_options)) {
2896*cf5a6c84SAndroid Build Coastguard Worker case 0: TT.singleline = 1;
2897*cf5a6c84SAndroid Build Coastguard Worker break;
2898*cf5a6c84SAndroid Build Coastguard Worker case 1: {
2899*cf5a6c84SAndroid Build Coastguard Worker if (isdigit(*ptr)) {
2900*cf5a6c84SAndroid Build Coastguard Worker long num = atolx(ptr);
2901*cf5a6c84SAndroid Build Coastguard Worker if (num == 4) TT.addressfamily = AF_INET;
2902*cf5a6c84SAndroid Build Coastguard Worker else if (num == 6) TT.addressfamily = AF_INET6;
2903*cf5a6c84SAndroid Build Coastguard Worker else TT.addressfamily = AF_PACKET;
2904*cf5a6c84SAndroid Build Coastguard Worker } else {
2905*cf5a6c84SAndroid Build Coastguard Worker struct arglist ip_aflist[] = {{"inet", AF_INET},
2906*cf5a6c84SAndroid Build Coastguard Worker {"inet6", AF_INET6}, {"link", AF_PACKET}, {NULL, -1}};
2907*cf5a6c84SAndroid Build Coastguard Worker
2908*cf5a6c84SAndroid Build Coastguard Worker if (!*++optargv) help_exit(0);
2909*cf5a6c84SAndroid Build Coastguard Worker if ((TT.addressfamily = string_to_idx(*optargv, ip_aflist)) == -1)
2910*cf5a6c84SAndroid Build Coastguard Worker error_exit("wrong family '%s'", *optargv);
2911*cf5a6c84SAndroid Build Coastguard Worker }
2912*cf5a6c84SAndroid Build Coastguard Worker }
2913*cf5a6c84SAndroid Build Coastguard Worker break;
2914*cf5a6c84SAndroid Build Coastguard Worker case 2:
2915*cf5a6c84SAndroid Build Coastguard Worker TT.stats++;
2916*cf5a6c84SAndroid Build Coastguard Worker break;
2917*cf5a6c84SAndroid Build Coastguard Worker default: help_exit(0);
2918*cf5a6c84SAndroid Build Coastguard Worker break; // unreachable code.
2919*cf5a6c84SAndroid Build Coastguard Worker }
2920*cf5a6c84SAndroid Build Coastguard Worker }
2921*cf5a6c84SAndroid Build Coastguard Worker
2922*cf5a6c84SAndroid Build Coastguard Worker TT.sockfd = xsocket(AF_NETLINK, SOCK_RAW, NETLINK_ROUTE);
2923*cf5a6c84SAndroid Build Coastguard Worker
2924*cf5a6c84SAndroid Build Coastguard Worker if (isip) {// only for ip
2925*cf5a6c84SAndroid Build Coastguard Worker if (*optargv) {
2926*cf5a6c84SAndroid Build Coastguard Worker struct arglist ip_objectlist[] = { {"address", 0}, {"link", 1},
2927*cf5a6c84SAndroid Build Coastguard Worker {"route", 2}, {"rule", 3}, {"tunnel", 4}, {"tunl", 4}, {NULL, -1}};
2928*cf5a6c84SAndroid Build Coastguard Worker
2929*cf5a6c84SAndroid Build Coastguard Worker if ((idx = substring_to_idx(*optargv, ip_objectlist)) == -1) help_exit(0);
2930*cf5a6c84SAndroid Build Coastguard Worker ipcmd = cmdobjlist[idx];
2931*cf5a6c84SAndroid Build Coastguard Worker toys.exitval = ipcmd(++optargv);
2932*cf5a6c84SAndroid Build Coastguard Worker } else help_exit(0);
2933*cf5a6c84SAndroid Build Coastguard Worker } else {
2934*cf5a6c84SAndroid Build Coastguard Worker struct arglist ip_objectlist[] = { {"ipaddr", 0}, {"iplink", 1},
2935*cf5a6c84SAndroid Build Coastguard Worker {"iproute", 2}, {"iprule", 3}, {"iptunnel", 4}, {NULL, -1}};
2936*cf5a6c84SAndroid Build Coastguard Worker if ((idx = string_to_idx(toys.which->name, ip_objectlist)) == -1)
2937*cf5a6c84SAndroid Build Coastguard Worker help_exit(0);
2938*cf5a6c84SAndroid Build Coastguard Worker ipcmd = cmdobjlist[idx];
2939*cf5a6c84SAndroid Build Coastguard Worker toys.exitval = ipcmd(optargv);
2940*cf5a6c84SAndroid Build Coastguard Worker }
2941*cf5a6c84SAndroid Build Coastguard Worker xclose(TT.sockfd);
2942*cf5a6c84SAndroid Build Coastguard Worker if (rtdsfield_init) free_alist(rt_dsfield);
2943*cf5a6c84SAndroid Build Coastguard Worker if (rtrealms_init) free_alist(rt_realms);
2944*cf5a6c84SAndroid Build Coastguard Worker if (rtscope_init) free_alist(rt_scope);
2945*cf5a6c84SAndroid Build Coastguard Worker if (rttable_init) free_alist(rt_tables);
2946*cf5a6c84SAndroid Build Coastguard Worker if (rtprotos_init) free_alist(rt_protos);
2947*cf5a6c84SAndroid Build Coastguard Worker }
2948