blob: cb750e628430270b48ef7d8519091cc4f44ff771 [file] [log] [blame]
Bernhard Reutner-Fischerd1d23a62006-01-12 12:08:46 +00001/* vi: set sw=4 ts=4: */
Glenn L McGrath9a2d2722002-11-10 01:33:55 +00002/*
3 * iproute.c "ip route".
4 *
Bernhard Reutner-Fischerab187822005-10-26 10:47:26 +00005 * Licensed under the GPL v2 or later, see the file LICENSE in this tarball.
Glenn L McGrath9a2d2722002-11-10 01:33:55 +00006 *
7 * Authors: Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru>
8 *
9 *
10 * Changes:
11 *
12 * Rani Assaf <rani@magic.metawire.com> 980929: resolve addresses
13 * Kunihiro Ishiguro <kunihiro@zebra.org> 001102: rtnh_ifindex was not initialized
14 */
15
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000016#include <sys/socket.h>
Glenn L McGrath275be872002-12-16 07:37:21 +000017
18#include <stdlib.h>
19#include <string.h>
Glenn L McGrath4a4c6772003-02-15 11:50:33 +000020#include <fcntl.h>
21#include <unistd.h>
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000022
23#include "rt_names.h"
24#include "utils.h"
Bernhard Reutner-Fischer9a67ca32006-04-02 21:14:19 +000025#include "ip_common.h"
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000026
Glenn L McGrath275be872002-12-16 07:37:21 +000027#include "libbb.h"
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000028
29#ifndef RTAX_RTTVAR
30#define RTAX_RTTVAR RTAX_HOPS
31#endif
32
33
34static struct
35{
36 int tb;
Glenn L McGrath4a4c6772003-02-15 11:50:33 +000037 int flushed;
38 char *flushb;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000039 int flushp;
40 int flushe;
41 struct rtnl_handle *rth;
42 int protocol, protocolmask;
43 int scope, scopemask;
44 int type, typemask;
45 int tos, tosmask;
46 int iif, iifmask;
47 int oif, oifmask;
48 int realm, realmmask;
49 inet_prefix rprefsrc;
50 inet_prefix rvia;
51 inet_prefix rdst;
52 inet_prefix mdst;
53 inet_prefix rsrc;
54 inet_prefix msrc;
55} filter;
56
Glenn L McGrath4a4c6772003-02-15 11:50:33 +000057static int flush_update(void)
58{
59 if (rtnl_send(filter.rth, filter.flushb, filter.flushp) < 0) {
60 perror("Failed to send flush request\n");
61 return -1;
62 }
63 filter.flushp = 0;
64 return 0;
65}
66
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000067static int print_route(struct sockaddr_nl *who ATTRIBUTE_UNUSED,
Bernhard Reutner-Fischerd1d23a62006-01-12 12:08:46 +000068 struct nlmsghdr *n, void *arg)
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000069{
70 FILE *fp = (FILE*)arg;
71 struct rtmsg *r = NLMSG_DATA(n);
72 int len = n->nlmsg_len;
73 struct rtattr * tb[RTA_MAX+1];
74 char abuf[256];
Glenn L McGrathfbf0b8a2003-04-26 02:22:19 +000075 inet_prefix dst;
76 inet_prefix src;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000077 int host_len = -1;
78 SPRINT_BUF(b1);
Eric Andersenc7bda1c2004-03-15 08:29:22 +000079
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000080
81 if (n->nlmsg_type != RTM_NEWROUTE && n->nlmsg_type != RTM_DELROUTE) {
82 fprintf(stderr, "Not a route: %08x %08x %08x\n",
83 n->nlmsg_len, n->nlmsg_type, n->nlmsg_flags);
84 return 0;
85 }
Glenn L McGrath4a4c6772003-02-15 11:50:33 +000086 if (filter.flushb && n->nlmsg_type != RTM_NEWROUTE)
87 return 0;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000088 len -= NLMSG_LENGTH(sizeof(*r));
89 if (len < 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +000090 bb_error_msg("wrong nlmsg len %d", len);
Glenn L McGrath9a2d2722002-11-10 01:33:55 +000091 return -1;
92 }
93
94 if (r->rtm_family == AF_INET6)
95 host_len = 128;
96 else if (r->rtm_family == AF_INET)
97 host_len = 32;
98
99 if (r->rtm_family == AF_INET6) {
100 if (filter.tb) {
101 if (filter.tb < 0) {
Glenn L McGrath16528552002-11-28 11:17:19 +0000102 if (!(r->rtm_flags&RTM_F_CLONED)) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000103 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000104 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000105 } else {
Glenn L McGrath16528552002-11-28 11:17:19 +0000106 if (r->rtm_flags&RTM_F_CLONED) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000107 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000108 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000109 if (filter.tb == RT_TABLE_LOCAL) {
Glenn L McGrath16528552002-11-28 11:17:19 +0000110 if (r->rtm_type != RTN_LOCAL) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000111 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000112 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000113 } else if (filter.tb == RT_TABLE_MAIN) {
Glenn L McGrath16528552002-11-28 11:17:19 +0000114 if (r->rtm_type == RTN_LOCAL) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000115 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000116 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000117 } else {
118 return 0;
119 }
120 }
121 }
122 } else {
Glenn L McGrath16528552002-11-28 11:17:19 +0000123 if (filter.tb > 0 && filter.tb != r->rtm_table) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000124 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000125 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000126 }
127 if (filter.rdst.family &&
Glenn L McGrath16528552002-11-28 11:17:19 +0000128 (r->rtm_family != filter.rdst.family || filter.rdst.bitlen > r->rtm_dst_len)) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000129 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000130 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000131 if (filter.mdst.family &&
132 (r->rtm_family != filter.mdst.family ||
Glenn L McGrath16528552002-11-28 11:17:19 +0000133 (filter.mdst.bitlen >= 0 && filter.mdst.bitlen < r->rtm_dst_len))) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000134 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000135 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000136 if (filter.rsrc.family &&
Glenn L McGrath16528552002-11-28 11:17:19 +0000137 (r->rtm_family != filter.rsrc.family || filter.rsrc.bitlen > r->rtm_src_len)) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000138 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000139 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000140 if (filter.msrc.family &&
141 (r->rtm_family != filter.msrc.family ||
Glenn L McGrath16528552002-11-28 11:17:19 +0000142 (filter.msrc.bitlen >= 0 && filter.msrc.bitlen < r->rtm_src_len))) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000143 return 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000144 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000145
146 memset(tb, 0, sizeof(tb));
147 parse_rtattr(tb, RTA_MAX, RTM_RTA(r), len);
148
Glenn L McGrathfbf0b8a2003-04-26 02:22:19 +0000149 if (filter.rdst.family && inet_addr_match(&dst, &filter.rdst, filter.rdst.bitlen))
150 return 0;
151 if (filter.mdst.family && filter.mdst.bitlen >= 0 &&
152 inet_addr_match(&dst, &filter.mdst, r->rtm_dst_len))
153 return 0;
154
155 if (filter.rsrc.family && inet_addr_match(&src, &filter.rsrc, filter.rsrc.bitlen))
156 return 0;
157 if (filter.msrc.family && filter.msrc.bitlen >= 0 &&
158 inet_addr_match(&src, &filter.msrc, r->rtm_src_len))
159 return 0;
160
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000161 if (filter.flushb &&
162 r->rtm_family == AF_INET6 &&
163 r->rtm_dst_len == 0 &&
164 r->rtm_type == RTN_UNREACHABLE &&
165 tb[RTA_PRIORITY] &&
166 *(int*)RTA_DATA(tb[RTA_PRIORITY]) == -1)
167 return 0;
168
169 if (filter.flushb) {
170 struct nlmsghdr *fn;
171 if (NLMSG_ALIGN(filter.flushp) + n->nlmsg_len > filter.flushe) {
172 if (flush_update())
173 return -1;
174 }
175 fn = (struct nlmsghdr*)(filter.flushb + NLMSG_ALIGN(filter.flushp));
176 memcpy(fn, n, n->nlmsg_len);
177 fn->nlmsg_type = RTM_DELROUTE;
178 fn->nlmsg_flags = NLM_F_REQUEST;
179 fn->nlmsg_seq = ++filter.rth->seq;
180 filter.flushp = (((char*)fn) + n->nlmsg_len) - filter.flushb;
181 filter.flushed++;
182 return 0;
183 }
184
Glenn L McGrath16528552002-11-28 11:17:19 +0000185 if (n->nlmsg_type == RTM_DELROUTE) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000186 fprintf(fp, "Deleted ");
Glenn L McGrath16528552002-11-28 11:17:19 +0000187 }
188 if (r->rtm_type != RTN_UNICAST && !filter.type) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000189 fprintf(fp, "%s ", rtnl_rtntype_n2a(r->rtm_type, b1, sizeof(b1)));
Glenn L McGrath16528552002-11-28 11:17:19 +0000190 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000191
192 if (tb[RTA_DST]) {
193 if (r->rtm_dst_len != host_len) {
194 fprintf(fp, "%s/%u ", rt_addr_n2a(r->rtm_family,
195 RTA_PAYLOAD(tb[RTA_DST]),
196 RTA_DATA(tb[RTA_DST]),
197 abuf, sizeof(abuf)),
198 r->rtm_dst_len
199 );
200 } else {
201 fprintf(fp, "%s ", format_host(r->rtm_family,
202 RTA_PAYLOAD(tb[RTA_DST]),
203 RTA_DATA(tb[RTA_DST]),
204 abuf, sizeof(abuf))
205 );
206 }
207 } else if (r->rtm_dst_len) {
208 fprintf(fp, "0/%d ", r->rtm_dst_len);
209 } else {
210 fprintf(fp, "default ");
211 }
212 if (tb[RTA_SRC]) {
213 if (r->rtm_src_len != host_len) {
214 fprintf(fp, "from %s/%u ", rt_addr_n2a(r->rtm_family,
215 RTA_PAYLOAD(tb[RTA_SRC]),
216 RTA_DATA(tb[RTA_SRC]),
217 abuf, sizeof(abuf)),
218 r->rtm_src_len
219 );
220 } else {
221 fprintf(fp, "from %s ", format_host(r->rtm_family,
222 RTA_PAYLOAD(tb[RTA_SRC]),
223 RTA_DATA(tb[RTA_SRC]),
224 abuf, sizeof(abuf))
225 );
226 }
227 } else if (r->rtm_src_len) {
228 fprintf(fp, "from 0/%u ", r->rtm_src_len);
229 }
230 if (tb[RTA_GATEWAY] && filter.rvia.bitlen != host_len) {
Eric Andersenc7bda1c2004-03-15 08:29:22 +0000231 fprintf(fp, "via %s ",
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000232 format_host(r->rtm_family,
233 RTA_PAYLOAD(tb[RTA_GATEWAY]),
234 RTA_DATA(tb[RTA_GATEWAY]),
235 abuf, sizeof(abuf)));
236 }
Glenn L McGrath16528552002-11-28 11:17:19 +0000237 if (tb[RTA_OIF] && filter.oifmask != -1) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000238 fprintf(fp, "dev %s ", ll_index_to_name(*(int*)RTA_DATA(tb[RTA_OIF])));
Glenn L McGrath16528552002-11-28 11:17:19 +0000239 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000240
241 if (tb[RTA_PREFSRC] && filter.rprefsrc.bitlen != host_len) {
242 /* Do not use format_host(). It is our local addr
243 and symbolic name will not be useful.
244 */
Eric Andersenc7bda1c2004-03-15 08:29:22 +0000245 fprintf(fp, " src %s ",
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000246 rt_addr_n2a(r->rtm_family,
247 RTA_PAYLOAD(tb[RTA_PREFSRC]),
248 RTA_DATA(tb[RTA_PREFSRC]),
249 abuf, sizeof(abuf)));
250 }
Glenn L McGrath16528552002-11-28 11:17:19 +0000251 if (tb[RTA_PRIORITY]) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000252 fprintf(fp, " metric %d ", *(__u32*)RTA_DATA(tb[RTA_PRIORITY]));
Glenn L McGrath16528552002-11-28 11:17:19 +0000253 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000254 if (r->rtm_family == AF_INET6) {
255 struct rta_cacheinfo *ci = NULL;
Glenn L McGrath16528552002-11-28 11:17:19 +0000256 if (tb[RTA_CACHEINFO]) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000257 ci = RTA_DATA(tb[RTA_CACHEINFO]);
Glenn L McGrath16528552002-11-28 11:17:19 +0000258 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000259 if ((r->rtm_flags & RTM_F_CLONED) || (ci && ci->rta_expires)) {
260 static int hz;
Glenn L McGrath16528552002-11-28 11:17:19 +0000261 if (!hz) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000262 hz = get_hz();
Glenn L McGrath16528552002-11-28 11:17:19 +0000263 }
264 if (r->rtm_flags & RTM_F_CLONED) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000265 fprintf(fp, "%s cache ", _SL_);
Glenn L McGrath16528552002-11-28 11:17:19 +0000266 }
267 if (ci->rta_expires) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000268 fprintf(fp, " expires %dsec", ci->rta_expires/hz);
Glenn L McGrath16528552002-11-28 11:17:19 +0000269 }
270 if (ci->rta_error != 0) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000271 fprintf(fp, " error %d", ci->rta_error);
Glenn L McGrath16528552002-11-28 11:17:19 +0000272 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000273 } else if (ci) {
274 if (ci->rta_error != 0)
275 fprintf(fp, " error %d", ci->rta_error);
276 }
277 }
278 if (tb[RTA_IIF] && filter.iifmask != -1) {
279 fprintf(fp, " iif %s", ll_index_to_name(*(int*)RTA_DATA(tb[RTA_IIF])));
280 }
281 fprintf(fp, "\n");
282 fflush(fp);
283 return 0;
284}
285
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000286static int iproute_modify(int cmd, unsigned flags, int argc, char **argv)
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000287{
288 struct rtnl_handle rth;
289 struct {
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000290 struct nlmsghdr n;
291 struct rtmsg r;
292 char buf[1024];
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000293 } req;
294 char mxbuf[256];
295 struct rtattr * mxrta = (void*)mxbuf;
296 unsigned mxlock = 0;
297 char *d = NULL;
298 int gw_ok = 0;
299 int dst_ok = 0;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000300 int proto_ok = 0;
301 int type_ok = 0;
302
303 memset(&req, 0, sizeof(req));
304
305 req.n.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg));
306 req.n.nlmsg_flags = NLM_F_REQUEST|flags;
307 req.n.nlmsg_type = cmd;
308 req.r.rtm_family = preferred_family;
309 req.r.rtm_table = RT_TABLE_MAIN;
310 req.r.rtm_scope = RT_SCOPE_NOWHERE;
311
312 if (cmd != RTM_DELROUTE) {
313 req.r.rtm_protocol = RTPROT_BOOT;
314 req.r.rtm_scope = RT_SCOPE_UNIVERSE;
315 req.r.rtm_type = RTN_UNICAST;
316 }
317
318 mxrta->rta_type = RTA_METRICS;
319 mxrta->rta_len = RTA_LENGTH(0);
320
321 while (argc > 0) {
322 if (strcmp(*argv, "src") == 0) {
323 inet_prefix addr;
324 NEXT_ARG();
325 get_addr(&addr, *argv, req.r.rtm_family);
Glenn L McGrath16528552002-11-28 11:17:19 +0000326 if (req.r.rtm_family == AF_UNSPEC) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000327 req.r.rtm_family = addr.family;
Glenn L McGrath16528552002-11-28 11:17:19 +0000328 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000329 addattr_l(&req.n, sizeof(req), RTA_PREFSRC, &addr.data, addr.bytelen);
330 } else if (strcmp(*argv, "via") == 0) {
331 inet_prefix addr;
332 gw_ok = 1;
333 NEXT_ARG();
334 get_addr(&addr, *argv, req.r.rtm_family);
Glenn L McGrath16528552002-11-28 11:17:19 +0000335 if (req.r.rtm_family == AF_UNSPEC) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000336 req.r.rtm_family = addr.family;
Glenn L McGrath16528552002-11-28 11:17:19 +0000337 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000338 addattr_l(&req.n, sizeof(req), RTA_GATEWAY, &addr.data, addr.bytelen);
339 } else if (strcmp(*argv, "mtu") == 0) {
340 unsigned mtu;
341 NEXT_ARG();
342 if (strcmp(*argv, "lock") == 0) {
343 mxlock |= (1<<RTAX_MTU);
344 NEXT_ARG();
345 }
Glenn L McGrath16528552002-11-28 11:17:19 +0000346 if (get_unsigned(&mtu, *argv, 0)) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000347 invarg("\"mtu\" value is invalid\n", *argv);
Glenn L McGrath16528552002-11-28 11:17:19 +0000348 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000349 rta_addattr32(mxrta, sizeof(mxbuf), RTAX_MTU, mtu);
350 } else if (matches(*argv, "protocol") == 0) {
Eric Andersend78aea82006-01-30 18:00:02 +0000351 uint32_t prot;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000352 NEXT_ARG();
353 if (rtnl_rtprot_a2n(&prot, *argv))
354 invarg("\"protocol\" value is invalid\n", *argv);
355 req.r.rtm_protocol = prot;
356 proto_ok =1;
357 } else if (strcmp(*argv, "dev") == 0 ||
358 strcmp(*argv, "oif") == 0) {
359 NEXT_ARG();
360 d = *argv;
361 } else {
362 int type;
363 inet_prefix dst;
364
365 if (strcmp(*argv, "to") == 0) {
366 NEXT_ARG();
367 }
368 if ((**argv < '0' || **argv > '9') &&
369 rtnl_rtntype_a2n(&type, *argv) == 0) {
370 NEXT_ARG();
371 req.r.rtm_type = type;
372 type_ok = 1;
373 }
374
Glenn L McGrath16528552002-11-28 11:17:19 +0000375 if (dst_ok) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000376 duparg2("to", *argv);
Glenn L McGrath16528552002-11-28 11:17:19 +0000377 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000378 get_prefix(&dst, *argv, req.r.rtm_family);
Glenn L McGrath16528552002-11-28 11:17:19 +0000379 if (req.r.rtm_family == AF_UNSPEC) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000380 req.r.rtm_family = dst.family;
Glenn L McGrath16528552002-11-28 11:17:19 +0000381 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000382 req.r.rtm_dst_len = dst.bitlen;
383 dst_ok = 1;
Glenn L McGrath16528552002-11-28 11:17:19 +0000384 if (dst.bytelen) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000385 addattr_l(&req.n, sizeof(req), RTA_DST, &dst.data, dst.bytelen);
Glenn L McGrath16528552002-11-28 11:17:19 +0000386 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000387 }
388 argc--; argv++;
389 }
390
Glenn L McGrath16528552002-11-28 11:17:19 +0000391 if (rtnl_open(&rth, 0) < 0) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000392 exit(1);
Glenn L McGrath16528552002-11-28 11:17:19 +0000393 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000394
Eric Andersen5780adb2002-11-15 09:12:47 +0000395 if (d) {
396 int idx;
397
398 ll_init_map(&rth);
399
400 if (d) {
401 if ((idx = ll_name_to_index(d)) == 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000402 bb_error_msg("Cannot find device \"%s\"", d);
Eric Andersen5780adb2002-11-15 09:12:47 +0000403 return -1;
404 }
405 addattr32(&req.n, sizeof(req), RTA_OIF, idx);
406 }
407 }
408
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000409 if (mxrta->rta_len > RTA_LENGTH(0)) {
Glenn L McGrath16528552002-11-28 11:17:19 +0000410 if (mxlock) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000411 rta_addattr32(mxrta, sizeof(mxbuf), RTAX_LOCK, mxlock);
Glenn L McGrath16528552002-11-28 11:17:19 +0000412 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000413 addattr_l(&req.n, sizeof(req), RTA_METRICS, RTA_DATA(mxrta), RTA_PAYLOAD(mxrta));
414 }
415
Glenn L McGrath16528552002-11-28 11:17:19 +0000416 if (req.r.rtm_family == AF_UNSPEC) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000417 req.r.rtm_family = AF_INET;
Glenn L McGrath16528552002-11-28 11:17:19 +0000418 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000419
Glenn L McGrath16528552002-11-28 11:17:19 +0000420 if (rtnl_talk(&rth, &req.n, 0, 0, NULL, NULL, NULL) < 0) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000421 exit(2);
Glenn L McGrath16528552002-11-28 11:17:19 +0000422 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000423
424 return 0;
425}
426
427static int rtnl_rtcache_request(struct rtnl_handle *rth, int family)
428{
429 struct {
430 struct nlmsghdr nlh;
431 struct rtmsg rtm;
432 } req;
433 struct sockaddr_nl nladdr;
434
435 memset(&nladdr, 0, sizeof(nladdr));
436 memset(&req, 0, sizeof(req));
437 nladdr.nl_family = AF_NETLINK;
438
439 req.nlh.nlmsg_len = sizeof(req);
440 req.nlh.nlmsg_type = RTM_GETROUTE;
441 req.nlh.nlmsg_flags = NLM_F_ROOT|NLM_F_REQUEST;
442 req.nlh.nlmsg_pid = 0;
443 req.nlh.nlmsg_seq = rth->dump = ++rth->seq;
444 req.rtm.rtm_family = family;
445 req.rtm.rtm_flags |= RTM_F_CLONED;
446
447 return sendto(rth->fd, (void*)&req, sizeof(req), 0, (struct sockaddr*)&nladdr, sizeof(nladdr));
448}
449
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000450static int iproute_flush_cache(void)
451{
452#define ROUTE_FLUSH_PATH "/proc/sys/net/ipv4/route/flush"
453
454 int len;
455 int flush_fd = open (ROUTE_FLUSH_PATH, O_WRONLY);
456 char *buffer = "-1";
457
458 if (flush_fd < 0) {
459 fprintf (stderr, "Cannot open \"%s\"\n", ROUTE_FLUSH_PATH);
460 return -1;
461 }
462
463 len = strlen (buffer);
464
465 if ((write (flush_fd, (void *)buffer, len)) < len) {
466 fprintf (stderr, "Cannot flush routing cache\n");
467 return -1;
468 }
469 close(flush_fd);
470 return 0;
471}
472
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000473static void iproute_reset_filter(void)
474{
475 memset(&filter, 0, sizeof(filter));
476 filter.mdst.bitlen = -1;
477 filter.msrc.bitlen = -1;
478}
479
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000480static int iproute_list_or_flush(int argc, char **argv, int flush)
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000481{
482 int do_ipv6 = preferred_family;
483 struct rtnl_handle rth;
484 char *id = NULL;
485 char *od = NULL;
486
487 iproute_reset_filter();
488 filter.tb = RT_TABLE_MAIN;
489
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000490 if (flush && argc <= 0) {
491 fprintf(stderr, "\"ip route flush\" requires arguments.\n");
492 return -1;
493 }
494
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000495 while (argc > 0) {
496 if (matches(*argv, "protocol") == 0) {
Eric Andersend78aea82006-01-30 18:00:02 +0000497 uint32_t prot = 0;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000498 NEXT_ARG();
499 filter.protocolmask = -1;
500 if (rtnl_rtprot_a2n(&prot, *argv)) {
Glenn L McGrath16528552002-11-28 11:17:19 +0000501 if (strcmp(*argv, "all") != 0) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000502 invarg("invalid \"protocol\"\n", *argv);
Glenn L McGrath16528552002-11-28 11:17:19 +0000503 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000504 prot = 0;
505 filter.protocolmask = 0;
506 }
507 filter.protocol = prot;
508 } else if (strcmp(*argv, "dev") == 0 ||
509 strcmp(*argv, "oif") == 0) {
510 NEXT_ARG();
511 od = *argv;
512 } else if (strcmp(*argv, "iif") == 0) {
513 NEXT_ARG();
514 id = *argv;
515 } else if (matches(*argv, "from") == 0) {
516 NEXT_ARG();
517 if (matches(*argv, "root") == 0) {
518 NEXT_ARG();
519 get_prefix(&filter.rsrc, *argv, do_ipv6);
520 } else if (matches(*argv, "match") == 0) {
521 NEXT_ARG();
522 get_prefix(&filter.msrc, *argv, do_ipv6);
523 } else {
524 if (matches(*argv, "exact") == 0) {
525 NEXT_ARG();
526 }
527 get_prefix(&filter.msrc, *argv, do_ipv6);
528 filter.rsrc = filter.msrc;
529 }
530 } else {
531 if (matches(*argv, "to") == 0) {
532 NEXT_ARG();
533 }
534 if (matches(*argv, "root") == 0) {
535 NEXT_ARG();
536 get_prefix(&filter.rdst, *argv, do_ipv6);
537 } else if (matches(*argv, "match") == 0) {
538 NEXT_ARG();
539 get_prefix(&filter.mdst, *argv, do_ipv6);
Paul Fox5dc0cee2005-07-20 19:01:05 +0000540 } else if (matches(*argv, "table") == 0) {
541 NEXT_ARG();
542 if (matches(*argv, "cache") == 0) {
543 filter.tb = -1;
544 } else if (matches(*argv, "main") != 0) {
545 invarg("invalid \"table\"", *argv);
546 }
547 } else if (matches(*argv, "cache") == 0) {
548 filter.tb = -1;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000549 } else {
550 if (matches(*argv, "exact") == 0) {
551 NEXT_ARG();
552 }
553 get_prefix(&filter.mdst, *argv, do_ipv6);
554 filter.rdst = filter.mdst;
555 }
556 }
557 argc--; argv++;
558 }
559
Glenn L McGrath16528552002-11-28 11:17:19 +0000560 if (do_ipv6 == AF_UNSPEC && filter.tb) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000561 do_ipv6 = AF_INET;
Glenn L McGrath16528552002-11-28 11:17:19 +0000562 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000563
Glenn L McGrath16528552002-11-28 11:17:19 +0000564 if (rtnl_open(&rth, 0) < 0) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000565 exit(1);
Glenn L McGrath16528552002-11-28 11:17:19 +0000566 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000567
568 ll_init_map(&rth);
569
570 if (id || od) {
571 int idx;
572
573 if (id) {
574 if ((idx = ll_name_to_index(id)) == 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000575 bb_error_msg("Cannot find device \"%s\"", id);
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000576 return -1;
577 }
578 filter.iif = idx;
579 filter.iifmask = -1;
580 }
581 if (od) {
582 if ((idx = ll_name_to_index(od)) == 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000583 bb_error_msg("Cannot find device \"%s\"", od);
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000584 }
585 filter.oif = idx;
586 filter.oifmask = -1;
587 }
588 }
589
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000590 if (flush) {
Bernhard Reutner-Fischerab187822005-10-26 10:47:26 +0000591 int _round = 0;
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000592 char flushb[4096-512];
593
594 if (filter.tb == -1) {
595 if (do_ipv6 != AF_INET6)
596 iproute_flush_cache();
597 if (do_ipv6 == AF_INET)
598 return 0;
599 }
600
601 filter.flushb = flushb;
602 filter.flushp = 0;
603 filter.flushe = sizeof(flushb);
604 filter.rth = &rth;
605
606 for (;;) {
607 if (rtnl_wilddump_request(&rth, do_ipv6, RTM_GETROUTE) < 0) {
608 perror("Cannot send dump request");
609 return -1;
610 }
611 filter.flushed = 0;
612 if (rtnl_dump_filter(&rth, print_route, stdout, NULL, NULL) < 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000613 bb_error_msg("Flush terminated\n");
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000614 return -1;
615 }
616 if (filter.flushed == 0) {
Bernhard Reutner-Fischerab187822005-10-26 10:47:26 +0000617 if (_round == 0) {
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000618 if (filter.tb != -1 || do_ipv6 == AF_INET6)
619 fprintf(stderr, "Nothing to flush.\n");
620 }
621 fflush(stdout);
622 return 0;
623 }
Bernhard Reutner-Fischerab187822005-10-26 10:47:26 +0000624 _round++;
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000625 if (flush_update() < 0)
626 exit(1);
627 }
628 }
629
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000630 if (filter.tb != -1) {
631 if (rtnl_wilddump_request(&rth, do_ipv6, RTM_GETROUTE) < 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000632 bb_perror_msg_and_die("Cannot send dump request");
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000633 }
634 } else {
635 if (rtnl_rtcache_request(&rth, do_ipv6) < 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000636 bb_perror_msg_and_die("Cannot send dump request");
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000637 }
638 }
639
640 if (rtnl_dump_filter(&rth, print_route, stdout, NULL, NULL) < 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000641 bb_error_msg_and_die("Dump terminated");
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000642 }
643
644 exit(0);
645}
646
647
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000648static int iproute_get(int argc, char **argv)
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000649{
650 struct rtnl_handle rth;
651 struct {
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000652 struct nlmsghdr n;
653 struct rtmsg r;
654 char buf[1024];
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000655 } req;
656 char *idev = NULL;
657 char *odev = NULL;
658 int connected = 0;
659 int from_ok = 0;
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000660 static const char * const options[] =
"Vladimir N. Oleynik"2f0a5f92005-12-06 12:00:39 +0000661 { "from", "iif", "oif", "dev", "notify", "connected", "to", 0 };
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000662
663 memset(&req, 0, sizeof(req));
664
665 iproute_reset_filter();
666
667 req.n.nlmsg_len = NLMSG_LENGTH(sizeof(struct rtmsg));
668 req.n.nlmsg_flags = NLM_F_REQUEST;
669 req.n.nlmsg_type = RTM_GETROUTE;
670 req.r.rtm_family = preferred_family;
671 req.r.rtm_table = 0;
672 req.r.rtm_protocol = 0;
673 req.r.rtm_scope = 0;
674 req.r.rtm_type = 0;
675 req.r.rtm_src_len = 0;
676 req.r.rtm_dst_len = 0;
677 req.r.rtm_tos = 0;
Eric Andersenc7bda1c2004-03-15 08:29:22 +0000678
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000679 while (argc > 0) {
Glenn L McGrath18eae002002-12-02 00:54:10 +0000680 switch (compare_string_array(options, *argv)) {
681 case 0: /* from */
682 {
683 inet_prefix addr;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000684 NEXT_ARG();
Glenn L McGrath18eae002002-12-02 00:54:10 +0000685 from_ok = 1;
686 get_prefix(&addr, *argv, req.r.rtm_family);
687 if (req.r.rtm_family == AF_UNSPEC) {
688 req.r.rtm_family = addr.family;
689 }
690 if (addr.bytelen) {
691 addattr_l(&req.n, sizeof(req), RTA_SRC, &addr.data, addr.bytelen);
692 }
693 req.r.rtm_src_len = addr.bitlen;
694 break;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000695 }
Glenn L McGrath18eae002002-12-02 00:54:10 +0000696 case 1: /* iif */
697 NEXT_ARG();
698 idev = *argv;
699 break;
700 case 2: /* oif */
701 case 3: /* dev */
702 NEXT_ARG();
703 odev = *argv;
704 break;
705 case 4: /* notify */
706 req.r.rtm_flags |= RTM_F_NOTIFY;
707 break;
708 case 5: /* connected */
709 connected = 1;
710 break;
711 case 6: /* to */
712 NEXT_ARG();
713 default:
714 {
715 inet_prefix addr;
716 get_prefix(&addr, *argv, req.r.rtm_family);
717 if (req.r.rtm_family == AF_UNSPEC) {
718 req.r.rtm_family = addr.family;
719 }
720 if (addr.bytelen) {
721 addattr_l(&req.n, sizeof(req), RTA_DST, &addr.data, addr.bytelen);
722 }
723 req.r.rtm_dst_len = addr.bitlen;
724 }
725 argc--; argv++;
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000726 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000727 }
728
729 if (req.r.rtm_dst_len == 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000730 bb_error_msg_and_die("need at least destination address");
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000731 }
732
733 if (rtnl_open(&rth, 0) < 0)
734 exit(1);
735
736 ll_init_map(&rth);
737
738 if (idev || odev) {
739 int idx;
740
741 if (idev) {
742 if ((idx = ll_name_to_index(idev)) == 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000743 bb_error_msg("Cannot find device \"%s\"", idev);
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000744 return -1;
745 }
746 addattr32(&req.n, sizeof(req), RTA_IIF, idx);
747 }
748 if (odev) {
749 if ((idx = ll_name_to_index(odev)) == 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000750 bb_error_msg("Cannot find device \"%s\"", odev);
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000751 return -1;
752 }
753 addattr32(&req.n, sizeof(req), RTA_OIF, idx);
754 }
755 }
756
Glenn L McGrath16528552002-11-28 11:17:19 +0000757 if (req.r.rtm_family == AF_UNSPEC) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000758 req.r.rtm_family = AF_INET;
Glenn L McGrath16528552002-11-28 11:17:19 +0000759 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000760
Glenn L McGrath16528552002-11-28 11:17:19 +0000761 if (rtnl_talk(&rth, &req.n, 0, 0, &req.n, NULL, NULL) < 0) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000762 exit(2);
Glenn L McGrath16528552002-11-28 11:17:19 +0000763 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000764
765 if (connected && !from_ok) {
766 struct rtmsg *r = NLMSG_DATA(&req.n);
767 int len = req.n.nlmsg_len;
768 struct rtattr * tb[RTA_MAX+1];
769
770 if (print_route(NULL, &req.n, (void*)stdout) < 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000771 bb_error_msg_and_die("An error :-)");
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000772 }
773
774 if (req.n.nlmsg_type != RTM_NEWROUTE) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000775 bb_error_msg("Not a route?");
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000776 return -1;
777 }
778 len -= NLMSG_LENGTH(sizeof(*r));
779 if (len < 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000780 bb_error_msg("Wrong len %d", len);
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000781 return -1;
782 }
783
784 memset(tb, 0, sizeof(tb));
785 parse_rtattr(tb, RTA_MAX, RTM_RTA(r), len);
786
787 if (tb[RTA_PREFSRC]) {
788 tb[RTA_PREFSRC]->rta_type = RTA_SRC;
789 r->rtm_src_len = 8*RTA_PAYLOAD(tb[RTA_PREFSRC]);
790 } else if (!tb[RTA_SRC]) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000791 bb_error_msg("Failed to connect the route");
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000792 return -1;
793 }
Glenn L McGrath16528552002-11-28 11:17:19 +0000794 if (!odev && tb[RTA_OIF]) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000795 tb[RTA_OIF]->rta_type = 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000796 }
797 if (tb[RTA_GATEWAY]) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000798 tb[RTA_GATEWAY]->rta_type = 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000799 }
800 if (!idev && tb[RTA_IIF]) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000801 tb[RTA_IIF]->rta_type = 0;
Glenn L McGrath16528552002-11-28 11:17:19 +0000802 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000803 req.n.nlmsg_flags = NLM_F_REQUEST;
804 req.n.nlmsg_type = RTM_GETROUTE;
805
Glenn L McGrath16528552002-11-28 11:17:19 +0000806 if (rtnl_talk(&rth, &req.n, 0, 0, &req.n, NULL, NULL) < 0) {
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000807 exit(2);
Glenn L McGrath16528552002-11-28 11:17:19 +0000808 }
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000809 }
810
811 if (print_route(NULL, &req.n, (void*)stdout) < 0) {
Manuel Novoa III cad53642003-03-19 09:13:01 +0000812 bb_error_msg_and_die("An error :-)");
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000813 }
814
815 exit(0);
816}
817
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000818int do_iproute(int argc, char **argv)
819{
Tim Rikerc1ef7bd2006-01-25 00:08:53 +0000820 static const char * const ip_route_commands[] =
821 { "add", "append", "change", "chg", "delete", "del", "get",
"Vladimir N. Oleynik"2f0a5f92005-12-06 12:00:39 +0000822 "list", "show", "prepend", "replace", "test", "flush", 0 };
823 int command_num = 7;
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000824 unsigned int flags = 0;
825 int cmd = RTM_NEWROUTE;
826
827 if (*argv) {
828 command_num = compare_string_array(ip_route_commands, *argv);
Glenn L McGrath16528552002-11-28 11:17:19 +0000829 }
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000830 switch(command_num) {
831 case 0: /* add*/
832 flags = NLM_F_CREATE|NLM_F_EXCL;
833 break;
834 case 1: /* append */
835 flags = NLM_F_CREATE|NLM_F_APPEND;
836 break;
837 case 2: /* change */
838 case 3: /* chg */
839 flags = NLM_F_REPLACE;
840 break;
841 case 4: /* delete */
Eric Andersen1a773a62003-12-19 10:47:40 +0000842 case 5: /* del */
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000843 cmd = RTM_DELROUTE;
844 break;
Eric Andersen1a773a62003-12-19 10:47:40 +0000845 case 6: /* get */
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000846 return iproute_get(argc-1, argv+1);
Eric Andersen1a773a62003-12-19 10:47:40 +0000847 case 7: /* list */
848 case 8: /* show */
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000849 return iproute_list_or_flush(argc-1, argv+1, 0);
Eric Andersen1a773a62003-12-19 10:47:40 +0000850 case 9: /* prepend */
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000851 flags = NLM_F_CREATE;
Eric Andersen1a773a62003-12-19 10:47:40 +0000852 case 10: /* replace */
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000853 flags = NLM_F_CREATE|NLM_F_REPLACE;
Eric Andersen1a773a62003-12-19 10:47:40 +0000854 case 11: /* test */
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000855 flags = NLM_F_EXCL;
Eric Andersen1a773a62003-12-19 10:47:40 +0000856 case 12: /* flush */
Glenn L McGrath4a4c6772003-02-15 11:50:33 +0000857 return iproute_list_or_flush(argc-1, argv+1, 1);
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000858 default:
Manuel Novoa III cad53642003-03-19 09:13:01 +0000859 bb_error_msg_and_die("Unknown command %s", *argv);
Glenn L McGrath16528552002-11-28 11:17:19 +0000860 }
Glenn L McGrathc82f2322002-12-02 00:35:23 +0000861
862 return iproute_modify(cmd, flags, argc-1, argv+1);
Glenn L McGrath9a2d2722002-11-10 01:33:55 +0000863}