blob: 0d9bbbd449a45f262cf62144571bad32c44b01aa [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/* Kernel routing table updates using netlink over GNU/Linux system.
2 * Copyright (C) 1997, 98, 99 Kunihiro Ishiguro
3 *
4 * This file is part of GNU Zebra.
5 *
6 * GNU Zebra is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation; either version 2, or (at your option) any
9 * later version.
10 *
11 * GNU Zebra is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License
17 * along with GNU Zebra; see the file COPYING. If not, write to the Free
18 * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
19 * 02111-1307, USA.
20 */
21
22#include <zebra.h>
23
24/* Hack for GNU libc version 2. */
25#ifndef MSG_TRUNC
26#define MSG_TRUNC 0x20
27#endif /* MSG_TRUNC */
28
29#include "linklist.h"
30#include "if.h"
31#include "log.h"
32#include "prefix.h"
33#include "connected.h"
34#include "table.h"
35#include "rib.h"
paule04ab742003-01-17 23:47:00 +000036#include "thread.h"
pauledd7c242003-06-04 13:59:38 +000037#include "privs.h"
paul718e3742002-12-13 20:15:29 +000038
39#include "zebra/zserv.h"
40#include "zebra/redistribute.h"
41#include "zebra/interface.h"
42#include "zebra/debug.h"
43
44/* Socket interface to kernel */
45struct nlsock
46{
47 int sock;
48 int seq;
49 struct sockaddr_nl snl;
hassofce954f2004-10-07 20:29:24 +000050 const char *name;
paul7021c422003-07-15 12:52:22 +000051} netlink = { -1, 0, {0}, "netlink-listen"}, /* kernel messages */
52 netlink_cmd = { -1, 0, {0}, "netlink-cmd"}, /* command channel */
53 netlink_addr = { -1, 0, {0}, "netlink-addr"}; /* address channel */
paul718e3742002-12-13 20:15:29 +000054
paul7021c422003-07-15 12:52:22 +000055struct message nlmsg_str[] = {
paul718e3742002-12-13 20:15:29 +000056 {RTM_NEWROUTE, "RTM_NEWROUTE"},
57 {RTM_DELROUTE, "RTM_DELROUTE"},
58 {RTM_GETROUTE, "RTM_GETROUTE"},
59 {RTM_NEWLINK, "RTM_NEWLINK"},
60 {RTM_DELLINK, "RTM_DELLINK"},
61 {RTM_GETLINK, "RTM_GETLINK"},
62 {RTM_NEWADDR, "RTM_NEWADDR"},
63 {RTM_DELADDR, "RTM_DELADDR"},
64 {RTM_GETADDR, "RTM_GETADDR"},
paul7021c422003-07-15 12:52:22 +000065 {0, NULL}
paul718e3742002-12-13 20:15:29 +000066};
67
hassofce954f2004-10-07 20:29:24 +000068const char *nexthop_types_desc[] =
paul7021c422003-07-15 12:52:22 +000069{
70 "none",
71 "Directly connected",
72 "Interface route",
73 "IPv4 nexthop",
74 "IPv4 nexthop with ifindex",
75 "IPv4 nexthop with ifname",
76 "IPv6 nexthop"
77 "IPv6 nexthop with ifindex",
78 "IPv6 nexthop with ifname",
79 "Null0 nexthop",
80};
81
82
paulb21b19c2003-06-15 01:28:29 +000083extern struct zebra_t zebrad;
paul718e3742002-12-13 20:15:29 +000084
pauledd7c242003-06-04 13:59:38 +000085extern struct zebra_privs_t zserv_privs;
86
hassoc34b6b52004-08-31 13:41:49 +000087extern u_int32_t nl_rcvbufsize;
88
paul718e3742002-12-13 20:15:29 +000089/* Make socket for Linux netlink interface. */
90static int
91netlink_socket (struct nlsock *nl, unsigned long groups)
92{
93 int ret;
94 struct sockaddr_nl snl;
95 int sock;
96 int namelen;
ajs4be019d2005-01-29 16:12:41 +000097 int save_errno;
paul718e3742002-12-13 20:15:29 +000098
99 sock = socket (AF_NETLINK, SOCK_RAW, NETLINK_ROUTE);
100 if (sock < 0)
101 {
102 zlog (NULL, LOG_ERR, "Can't open %s socket: %s", nl->name,
ajs6099b3b2004-11-20 02:06:59 +0000103 safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000104 return -1;
105 }
106
107 ret = fcntl (sock, F_SETFL, O_NONBLOCK);
108 if (ret < 0)
109 {
110 zlog (NULL, LOG_ERR, "Can't set %s socket flags: %s", nl->name,
ajs6099b3b2004-11-20 02:06:59 +0000111 safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000112 close (sock);
113 return -1;
114 }
paul7021c422003-07-15 12:52:22 +0000115
hassoc34b6b52004-08-31 13:41:49 +0000116 /* Set receive buffer size if it's set from command line */
117 if (nl_rcvbufsize)
118 {
119 u_int32_t oldsize, oldlen;
120 u_int32_t newsize, newlen;
121
122 oldlen = sizeof(oldsize);
123 newlen = sizeof(newsize);
124
125 ret = getsockopt(sock, SOL_SOCKET, SO_RCVBUF, &oldsize, &oldlen);
126 if (ret < 0)
127 {
128 zlog (NULL, LOG_ERR, "Can't get %s receive buffer size: %s", nl->name,
ajs6099b3b2004-11-20 02:06:59 +0000129 safe_strerror (errno));
hassoc34b6b52004-08-31 13:41:49 +0000130 close (sock);
131 return -1;
132 }
133
134 ret = setsockopt(sock, SOL_SOCKET, SO_RCVBUF, &nl_rcvbufsize,
135 sizeof(nl_rcvbufsize));
136 if (ret < 0)
137 {
138 zlog (NULL, LOG_ERR, "Can't set %s receive buffer size: %s", nl->name,
ajs6099b3b2004-11-20 02:06:59 +0000139 safe_strerror (errno));
hassoc34b6b52004-08-31 13:41:49 +0000140 close (sock);
141 return -1;
142 }
143
144 ret = getsockopt(sock, SOL_SOCKET, SO_RCVBUF, &newsize, &newlen);
145 if (ret < 0)
146 {
147 zlog (NULL, LOG_ERR, "Can't get %s receive buffer size: %s", nl->name,
ajs6099b3b2004-11-20 02:06:59 +0000148 safe_strerror (errno));
hassoc34b6b52004-08-31 13:41:49 +0000149 close (sock);
150 return -1;
151 }
152
153 zlog (NULL, LOG_INFO,
154 "Setting netlink socket receive buffer size: %u -> %u",
155 oldsize, newsize);
156 }
157
paul718e3742002-12-13 20:15:29 +0000158 memset (&snl, 0, sizeof snl);
159 snl.nl_family = AF_NETLINK;
160 snl.nl_groups = groups;
161
162 /* Bind the socket to the netlink structure for anything. */
paul7021c422003-07-15 12:52:22 +0000163 if (zserv_privs.change (ZPRIVS_RAISE))
164 {
165 zlog (NULL, LOG_ERR, "Can't raise privileges");
166 return -1;
167 }
pauledd7c242003-06-04 13:59:38 +0000168
paul718e3742002-12-13 20:15:29 +0000169 ret = bind (sock, (struct sockaddr *) &snl, sizeof snl);
ajs4be019d2005-01-29 16:12:41 +0000170 save_errno = errno;
hasso55e7ecd2004-08-06 08:41:56 +0000171 if (zserv_privs.change (ZPRIVS_LOWER))
172 zlog (NULL, LOG_ERR, "Can't lower privileges");
173
paul718e3742002-12-13 20:15:29 +0000174 if (ret < 0)
175 {
paul7021c422003-07-15 12:52:22 +0000176 zlog (NULL, LOG_ERR, "Can't bind %s socket to group 0x%x: %s",
ajs4be019d2005-01-29 16:12:41 +0000177 nl->name, snl.nl_groups, safe_strerror (save_errno));
paul718e3742002-12-13 20:15:29 +0000178 close (sock);
179 return -1;
180 }
paul7021c422003-07-15 12:52:22 +0000181
paul718e3742002-12-13 20:15:29 +0000182 /* multiple netlink sockets will have different nl_pid */
183 namelen = sizeof snl;
hassoc9e52be2004-09-26 16:09:34 +0000184 ret = getsockname (sock, (struct sockaddr *) &snl, (socklen_t *) &namelen);
paul718e3742002-12-13 20:15:29 +0000185 if (ret < 0 || namelen != sizeof snl)
186 {
187 zlog (NULL, LOG_ERR, "Can't get %s socket name: %s", nl->name,
ajs6099b3b2004-11-20 02:06:59 +0000188 safe_strerror (errno));
paul718e3742002-12-13 20:15:29 +0000189 close (sock);
190 return -1;
191 }
192
193 nl->snl = snl;
194 nl->sock = sock;
195 return ret;
196}
197
paul7021c422003-07-15 12:52:22 +0000198int
199set_netlink_blocking (struct nlsock *nl, int *flags)
paul5f37d862003-04-19 00:11:28 +0000200{
201
202 /* Change socket flags for blocking I/O. */
paul7021c422003-07-15 12:52:22 +0000203 if ((*flags = fcntl (nl->sock, F_GETFL, 0)) < 0)
paul5f37d862003-04-19 00:11:28 +0000204 {
paul7021c422003-07-15 12:52:22 +0000205 zlog (NULL, LOG_ERR, "%s:%i F_GETFL error: %s",
ajs6099b3b2004-11-20 02:06:59 +0000206 __FUNCTION__, __LINE__, safe_strerror (errno));
paul5f37d862003-04-19 00:11:28 +0000207 return -1;
208 }
209 *flags &= ~O_NONBLOCK;
paul7021c422003-07-15 12:52:22 +0000210 if (fcntl (nl->sock, F_SETFL, *flags) < 0)
paul5f37d862003-04-19 00:11:28 +0000211 {
paul7021c422003-07-15 12:52:22 +0000212 zlog (NULL, LOG_ERR, "%s:%i F_SETFL error: %s",
ajs6099b3b2004-11-20 02:06:59 +0000213 __FUNCTION__, __LINE__, safe_strerror (errno));
paul5f37d862003-04-19 00:11:28 +0000214 return -1;
215 }
216 return 0;
217}
218
paul7021c422003-07-15 12:52:22 +0000219int
220set_netlink_nonblocking (struct nlsock *nl, int *flags)
221{
paul5f37d862003-04-19 00:11:28 +0000222 /* Restore socket flags for nonblocking I/O */
223 *flags |= O_NONBLOCK;
paul7021c422003-07-15 12:52:22 +0000224 if (fcntl (nl->sock, F_SETFL, *flags) < 0)
paul5f37d862003-04-19 00:11:28 +0000225 {
paul7021c422003-07-15 12:52:22 +0000226 zlog (NULL, LOG_ERR, "%s:%i F_SETFL error: %s",
ajs6099b3b2004-11-20 02:06:59 +0000227 __FUNCTION__, __LINE__, safe_strerror (errno));
paul5f37d862003-04-19 00:11:28 +0000228 return -1;
229 }
230 return 0;
231}
232
paul718e3742002-12-13 20:15:29 +0000233/* Get type specified information from netlink. */
234static int
235netlink_request (int family, int type, struct nlsock *nl)
236{
237 int ret;
238 struct sockaddr_nl snl;
ajs4be019d2005-01-29 16:12:41 +0000239 int save_errno;
paul718e3742002-12-13 20:15:29 +0000240
241 struct
242 {
243 struct nlmsghdr nlh;
244 struct rtgenmsg g;
245 } req;
246
247
248 /* Check netlink socket. */
249 if (nl->sock < 0)
250 {
251 zlog (NULL, LOG_ERR, "%s socket isn't active.", nl->name);
252 return -1;
253 }
254
255 memset (&snl, 0, sizeof snl);
256 snl.nl_family = AF_NETLINK;
257
258 req.nlh.nlmsg_len = sizeof req;
259 req.nlh.nlmsg_type = type;
260 req.nlh.nlmsg_flags = NLM_F_ROOT | NLM_F_MATCH | NLM_F_REQUEST;
261 req.nlh.nlmsg_pid = 0;
262 req.nlh.nlmsg_seq = ++nl->seq;
263 req.g.rtgen_family = family;
pauledd7c242003-06-04 13:59:38 +0000264
265 /* linux appears to check capabilities on every message
266 * have to raise caps for every message sent
267 */
paul7021c422003-07-15 12:52:22 +0000268 if (zserv_privs.change (ZPRIVS_RAISE))
pauledd7c242003-06-04 13:59:38 +0000269 {
270 zlog (NULL, LOG_ERR, "Can't raise privileges");
271 return -1;
272 }
paul7021c422003-07-15 12:52:22 +0000273
274 ret = sendto (nl->sock, (void *) &req, sizeof req, 0,
275 (struct sockaddr *) &snl, sizeof snl);
ajs4be019d2005-01-29 16:12:41 +0000276 save_errno = errno;
paul7021c422003-07-15 12:52:22 +0000277
278 if (zserv_privs.change (ZPRIVS_LOWER))
279 zlog (NULL, LOG_ERR, "Can't lower privileges");
280
paul718e3742002-12-13 20:15:29 +0000281 if (ret < 0)
paul7021c422003-07-15 12:52:22 +0000282 {
283 zlog (NULL, LOG_ERR, "%s sendto failed: %s", nl->name,
ajs4be019d2005-01-29 16:12:41 +0000284 safe_strerror (save_errno));
paul718e3742002-12-13 20:15:29 +0000285 return -1;
286 }
pauledd7c242003-06-04 13:59:38 +0000287
paul718e3742002-12-13 20:15:29 +0000288 return 0;
289}
290
291/* Receive message from netlink interface and pass those information
292 to the given function. */
293static int
294netlink_parse_info (int (*filter) (struct sockaddr_nl *, struct nlmsghdr *),
paul7021c422003-07-15 12:52:22 +0000295 struct nlsock *nl)
paul718e3742002-12-13 20:15:29 +0000296{
297 int status;
298 int ret = 0;
299 int error;
300
301 while (1)
302 {
303 char buf[4096];
304 struct iovec iov = { buf, sizeof buf };
305 struct sockaddr_nl snl;
paul7021c422003-07-15 12:52:22 +0000306 struct msghdr msg = { (void *) &snl, sizeof snl, &iov, 1, NULL, 0, 0 };
paul718e3742002-12-13 20:15:29 +0000307 struct nlmsghdr *h;
ajs4be019d2005-01-29 16:12:41 +0000308 int save_errno;
paul718e3742002-12-13 20:15:29 +0000309
paul7021c422003-07-15 12:52:22 +0000310 if (zserv_privs.change (ZPRIVS_RAISE))
pauledd7c242003-06-04 13:59:38 +0000311 zlog (NULL, LOG_ERR, "Can't raise privileges");
paul7021c422003-07-15 12:52:22 +0000312
paul718e3742002-12-13 20:15:29 +0000313 status = recvmsg (nl->sock, &msg, 0);
ajs4be019d2005-01-29 16:12:41 +0000314 save_errno = errno;
paul7021c422003-07-15 12:52:22 +0000315
316 if (zserv_privs.change (ZPRIVS_LOWER))
pauledd7c242003-06-04 13:59:38 +0000317 zlog (NULL, LOG_ERR, "Can't lower privileges");
paul718e3742002-12-13 20:15:29 +0000318
319 if (status < 0)
paul7021c422003-07-15 12:52:22 +0000320 {
ajs4be019d2005-01-29 16:12:41 +0000321 if (save_errno == EINTR)
paul7021c422003-07-15 12:52:22 +0000322 continue;
ajs4be019d2005-01-29 16:12:41 +0000323 if (save_errno == EWOULDBLOCK || save_errno == EAGAIN)
paul7021c422003-07-15 12:52:22 +0000324 break;
ajs4be019d2005-01-29 16:12:41 +0000325 zlog (NULL, LOG_ERR, "%s recvmsg overrun: %s",
326 nl->name, safe_strerror(save_errno));
paul7021c422003-07-15 12:52:22 +0000327 continue;
328 }
paul718e3742002-12-13 20:15:29 +0000329
330 if (status == 0)
paul7021c422003-07-15 12:52:22 +0000331 {
332 zlog (NULL, LOG_ERR, "%s EOF", nl->name);
333 return -1;
334 }
paul718e3742002-12-13 20:15:29 +0000335
336 if (msg.msg_namelen != sizeof snl)
paul7021c422003-07-15 12:52:22 +0000337 {
338 zlog (NULL, LOG_ERR, "%s sender address length error: length %d",
339 nl->name, msg.msg_namelen);
340 return -1;
341 }
paulb84d3a12003-11-17 10:31:01 +0000342
343 /* JF: Ignore messages that aren't from the kernel */
344 if ( snl.nl_pid != 0 )
345 {
346 zlog ( NULL, LOG_ERR, "Ignoring message from pid %u", snl.nl_pid );
347 continue;
348 }
paul718e3742002-12-13 20:15:29 +0000349
paul7021c422003-07-15 12:52:22 +0000350 for (h = (struct nlmsghdr *) buf; NLMSG_OK (h, status);
351 h = NLMSG_NEXT (h, status))
352 {
353 /* Finish of reading. */
354 if (h->nlmsg_type == NLMSG_DONE)
355 return ret;
paul718e3742002-12-13 20:15:29 +0000356
paul7021c422003-07-15 12:52:22 +0000357 /* Error handling. */
358 if (h->nlmsg_type == NLMSG_ERROR)
359 {
360 struct nlmsgerr *err = (struct nlmsgerr *) NLMSG_DATA (h);
361
paul718e3742002-12-13 20:15:29 +0000362 /* If the error field is zero, then this is an ACK */
paul7021c422003-07-15 12:52:22 +0000363 if (err->error == 0)
paul718e3742002-12-13 20:15:29 +0000364 {
paul7021c422003-07-15 12:52:22 +0000365 if (IS_ZEBRA_DEBUG_KERNEL)
366 {
ajsb6178002004-12-07 21:12:56 +0000367 zlog_debug ("%s: %s ACK: type=%s(%u), seq=%u, pid=%d",
paul7021c422003-07-15 12:52:22 +0000368 __FUNCTION__, nl->name,
369 lookup (nlmsg_str, err->msg.nlmsg_type),
370 err->msg.nlmsg_type, err->msg.nlmsg_seq,
371 err->msg.nlmsg_pid);
paul718e3742002-12-13 20:15:29 +0000372 }
paul7021c422003-07-15 12:52:22 +0000373
374 /* return if not a multipart message, otherwise continue */
375 if (!(h->nlmsg_flags & NLM_F_MULTI))
376 {
377 return 0;
paul718e3742002-12-13 20:15:29 +0000378 }
paul7021c422003-07-15 12:52:22 +0000379 continue;
paul718e3742002-12-13 20:15:29 +0000380 }
paul7021c422003-07-15 12:52:22 +0000381
paul718e3742002-12-13 20:15:29 +0000382 if (h->nlmsg_len < NLMSG_LENGTH (sizeof (struct nlmsgerr)))
paul7021c422003-07-15 12:52:22 +0000383 {
384 zlog (NULL, LOG_ERR, "%s error: message truncated",
385 nl->name);
386 return -1;
387 }
pauld753e9e2003-01-22 19:45:50 +0000388
paul7021c422003-07-15 12:52:22 +0000389 /* Deal with Error Noise - MAG */
390 {
391 int loglvl = LOG_ERR;
392 int errnum = err->error;
393 int msg_type = err->msg.nlmsg_type;
paul718e3742002-12-13 20:15:29 +0000394
paul7021c422003-07-15 12:52:22 +0000395 if (nl == &netlink_cmd
396 && (-errnum == ENODEV || -errnum == ESRCH)
397 && (msg_type == RTM_NEWROUTE || msg_type == RTM_DELROUTE))
398 loglvl = LOG_DEBUG;
paul718e3742002-12-13 20:15:29 +0000399
paul7021c422003-07-15 12:52:22 +0000400 zlog (NULL, loglvl, "%s error: %s, type=%s(%u), "
401 "seq=%u, pid=%d",
ajs6099b3b2004-11-20 02:06:59 +0000402 nl->name, safe_strerror (-errnum),
paul7021c422003-07-15 12:52:22 +0000403 lookup (nlmsg_str, msg_type),
404 msg_type, err->msg.nlmsg_seq, err->msg.nlmsg_pid);
405 }
406 /*
407 ret = -1;
408 continue;
409 */
410 return -1;
411 }
paul718e3742002-12-13 20:15:29 +0000412
paul7021c422003-07-15 12:52:22 +0000413 /* OK we got netlink message. */
414 if (IS_ZEBRA_DEBUG_KERNEL)
ajsb6178002004-12-07 21:12:56 +0000415 zlog_debug ("netlink_parse_info: %s type %s(%u), seq=%u, pid=%d",
paul7021c422003-07-15 12:52:22 +0000416 nl->name,
417 lookup (nlmsg_str, h->nlmsg_type), h->nlmsg_type,
418 h->nlmsg_seq, h->nlmsg_pid);
419
420 /* skip unsolicited messages originating from command socket */
421 if (nl != &netlink_cmd && h->nlmsg_pid == netlink_cmd.snl.nl_pid)
422 {
423 if (IS_ZEBRA_DEBUG_KERNEL)
ajsb6178002004-12-07 21:12:56 +0000424 zlog_debug ("netlink_parse_info: %s packet comes from %s",
paul7021c422003-07-15 12:52:22 +0000425 nl->name, netlink_cmd.name);
426 continue;
427 }
428
429 error = (*filter) (&snl, h);
430 if (error < 0)
431 {
432 zlog (NULL, LOG_ERR, "%s filter function error", nl->name);
433 ret = error;
434 }
435 }
paul718e3742002-12-13 20:15:29 +0000436
437 /* After error care. */
438 if (msg.msg_flags & MSG_TRUNC)
paul7021c422003-07-15 12:52:22 +0000439 {
440 zlog (NULL, LOG_ERR, "%s error: message truncated", nl->name);
441 continue;
442 }
paul718e3742002-12-13 20:15:29 +0000443 if (status)
paul7021c422003-07-15 12:52:22 +0000444 {
445 zlog (NULL, LOG_ERR, "%s error: data remnant size %d", nl->name,
446 status);
447 return -1;
448 }
paul718e3742002-12-13 20:15:29 +0000449 }
450 return ret;
451}
452
453/* Utility function for parse rtattr. */
454static void
paul7021c422003-07-15 12:52:22 +0000455netlink_parse_rtattr (struct rtattr **tb, int max, struct rtattr *rta,
456 int len)
paul718e3742002-12-13 20:15:29 +0000457{
paul7021c422003-07-15 12:52:22 +0000458 while (RTA_OK (rta, len))
paul718e3742002-12-13 20:15:29 +0000459 {
460 if (rta->rta_type <= max)
paul7021c422003-07-15 12:52:22 +0000461 tb[rta->rta_type] = rta;
462 rta = RTA_NEXT (rta, len);
paul718e3742002-12-13 20:15:29 +0000463 }
464}
465
466/* Called from interface_lookup_netlink(). This function is only used
467 during bootstrap. */
468int
469netlink_interface (struct sockaddr_nl *snl, struct nlmsghdr *h)
470{
471 int len;
472 struct ifinfomsg *ifi;
473 struct rtattr *tb[IFLA_MAX + 1];
474 struct interface *ifp;
475 char *name;
476 int i;
477
478 ifi = NLMSG_DATA (h);
479
480 if (h->nlmsg_type != RTM_NEWLINK)
481 return 0;
482
483 len = h->nlmsg_len - NLMSG_LENGTH (sizeof (struct ifinfomsg));
484 if (len < 0)
485 return -1;
486
487 /* Looking up interface name. */
488 memset (tb, 0, sizeof tb);
489 netlink_parse_rtattr (tb, IFLA_MAX, IFLA_RTA (ifi), len);
paulc15cb242005-01-24 09:05:27 +0000490
491 /* check for wireless messages to ignore */
492 if ((tb[IFLA_WIRELESS] != NULL) && (ifi->ifi_change == 0))
493 {
494 if (IS_ZEBRA_DEBUG_KERNEL)
495 zlog_debug ("%s: ignoring IFLA_WIRELESS message", __func__);
496 return 0;
497 }
498
paul718e3742002-12-13 20:15:29 +0000499 if (tb[IFLA_IFNAME] == NULL)
500 return -1;
paul7021c422003-07-15 12:52:22 +0000501 name = (char *) RTA_DATA (tb[IFLA_IFNAME]);
paul718e3742002-12-13 20:15:29 +0000502
503 /* Add interface. */
504 ifp = if_get_by_name (name);
paul7021c422003-07-15 12:52:22 +0000505
paul718e3742002-12-13 20:15:29 +0000506 ifp->ifindex = ifi->ifi_index;
507 ifp->flags = ifi->ifi_flags & 0x0000fffff;
paul44145db2004-05-09 11:00:23 +0000508 ifp->mtu6 = ifp->mtu = *(int *) RTA_DATA (tb[IFLA_MTU]);
paul718e3742002-12-13 20:15:29 +0000509 ifp->metric = 1;
510
511 /* Hardware type and address. */
512 ifp->hw_type = ifi->ifi_type;
513
514 if (tb[IFLA_ADDRESS])
515 {
516 int hw_addr_len;
517
paul7021c422003-07-15 12:52:22 +0000518 hw_addr_len = RTA_PAYLOAD (tb[IFLA_ADDRESS]);
paul718e3742002-12-13 20:15:29 +0000519
520 if (hw_addr_len > INTERFACE_HWADDR_MAX)
paul7021c422003-07-15 12:52:22 +0000521 zlog_warn ("Hardware address is too large: %d", hw_addr_len);
paul718e3742002-12-13 20:15:29 +0000522 else
paul7021c422003-07-15 12:52:22 +0000523 {
524 ifp->hw_addr_len = hw_addr_len;
525 memcpy (ifp->hw_addr, RTA_DATA (tb[IFLA_ADDRESS]), hw_addr_len);
paul718e3742002-12-13 20:15:29 +0000526
paul7021c422003-07-15 12:52:22 +0000527 for (i = 0; i < hw_addr_len; i++)
528 if (ifp->hw_addr[i] != 0)
529 break;
paul718e3742002-12-13 20:15:29 +0000530
paul7021c422003-07-15 12:52:22 +0000531 if (i == hw_addr_len)
532 ifp->hw_addr_len = 0;
533 else
534 ifp->hw_addr_len = hw_addr_len;
535 }
paul718e3742002-12-13 20:15:29 +0000536 }
537
538 if_add_update (ifp);
539
540 return 0;
541}
542
543/* Lookup interface IPv4/IPv6 address. */
544int
545netlink_interface_addr (struct sockaddr_nl *snl, struct nlmsghdr *h)
546{
547 int len;
548 struct ifaddrmsg *ifa;
paul7021c422003-07-15 12:52:22 +0000549 struct rtattr *tb[IFA_MAX + 1];
paul718e3742002-12-13 20:15:29 +0000550 struct interface *ifp;
551 void *addr = NULL;
552 void *broad = NULL;
553 u_char flags = 0;
554 char *label = NULL;
555
556 ifa = NLMSG_DATA (h);
557
paul7021c422003-07-15 12:52:22 +0000558 if (ifa->ifa_family != AF_INET
paul718e3742002-12-13 20:15:29 +0000559#ifdef HAVE_IPV6
560 && ifa->ifa_family != AF_INET6
561#endif /* HAVE_IPV6 */
paul7021c422003-07-15 12:52:22 +0000562 )
paul718e3742002-12-13 20:15:29 +0000563 return 0;
564
565 if (h->nlmsg_type != RTM_NEWADDR && h->nlmsg_type != RTM_DELADDR)
566 return 0;
567
paul7021c422003-07-15 12:52:22 +0000568 len = h->nlmsg_len - NLMSG_LENGTH (sizeof (struct ifaddrmsg));
paul718e3742002-12-13 20:15:29 +0000569 if (len < 0)
570 return -1;
571
572 memset (tb, 0, sizeof tb);
573 netlink_parse_rtattr (tb, IFA_MAX, IFA_RTA (ifa), len);
574
575 ifp = if_lookup_by_index (ifa->ifa_index);
576 if (ifp == NULL)
577 {
578 zlog_err ("netlink_interface_addr can't find interface by index %d",
paul7021c422003-07-15 12:52:22 +0000579 ifa->ifa_index);
paul718e3742002-12-13 20:15:29 +0000580 return -1;
581 }
582
paul7021c422003-07-15 12:52:22 +0000583 if (IS_ZEBRA_DEBUG_KERNEL) /* remove this line to see initial ifcfg */
paul718e3742002-12-13 20:15:29 +0000584 {
paul00df0c12002-12-13 21:07:36 +0000585 char buf[BUFSIZ];
ajsb6178002004-12-07 21:12:56 +0000586 zlog_debug ("netlink_interface_addr %s %s/%d:",
paul7021c422003-07-15 12:52:22 +0000587 lookup (nlmsg_str, h->nlmsg_type),
588 ifp->name, ifa->ifa_prefixlen);
paul718e3742002-12-13 20:15:29 +0000589 if (tb[IFA_LOCAL])
ajsb6178002004-12-07 21:12:56 +0000590 zlog_debug (" IFA_LOCAL %s", inet_ntop (ifa->ifa_family,
paul7021c422003-07-15 12:52:22 +0000591 RTA_DATA (tb[IFA_LOCAL]),
592 buf, BUFSIZ));
paul718e3742002-12-13 20:15:29 +0000593 if (tb[IFA_ADDRESS])
ajsb6178002004-12-07 21:12:56 +0000594 zlog_debug (" IFA_ADDRESS %s", inet_ntop (ifa->ifa_family,
paul7021c422003-07-15 12:52:22 +0000595 RTA_DATA (tb
596 [IFA_ADDRESS]),
597 buf, BUFSIZ));
paul718e3742002-12-13 20:15:29 +0000598 if (tb[IFA_BROADCAST])
ajsb6178002004-12-07 21:12:56 +0000599 zlog_debug (" IFA_BROADCAST %s", inet_ntop (ifa->ifa_family,
paul7021c422003-07-15 12:52:22 +0000600 RTA_DATA (tb
601 [IFA_BROADCAST]),
602 buf, BUFSIZ));
paul00df0c12002-12-13 21:07:36 +0000603 if (tb[IFA_LABEL] && strcmp (ifp->name, RTA_DATA (tb[IFA_LABEL])))
ajsb6178002004-12-07 21:12:56 +0000604 zlog_debug (" IFA_LABEL %s", (char *)RTA_DATA (tb[IFA_LABEL]));
paul718e3742002-12-13 20:15:29 +0000605 }
paul31a476c2003-09-29 19:54:53 +0000606
607 if (tb[IFA_ADDRESS] == NULL)
608 tb[IFA_ADDRESS] = tb[IFA_LOCAL];
609
610 if (ifp->flags & IFF_POINTOPOINT)
paul7021c422003-07-15 12:52:22 +0000611 {
paul31a476c2003-09-29 19:54:53 +0000612 if (tb[IFA_LOCAL])
613 {
614 addr = RTA_DATA (tb[IFA_LOCAL]);
hasso3fb9cd62004-10-19 19:44:43 +0000615 if (tb[IFA_ADDRESS] &&
616 memcmp(RTA_DATA(tb[IFA_ADDRESS]),RTA_DATA(tb[IFA_LOCAL]),4))
617 /* if IFA_ADDRESS != IFA_LOCAL, then it's the peer address */
paul31a476c2003-09-29 19:54:53 +0000618 broad = RTA_DATA (tb[IFA_ADDRESS]);
619 else
620 broad = NULL;
621 }
622 else
623 {
624 if (tb[IFA_ADDRESS])
625 addr = RTA_DATA (tb[IFA_ADDRESS]);
626 else
627 addr = NULL;
628 }
paul7021c422003-07-15 12:52:22 +0000629 }
paul31a476c2003-09-29 19:54:53 +0000630 else
paul7021c422003-07-15 12:52:22 +0000631 {
paul31a476c2003-09-29 19:54:53 +0000632 if (tb[IFA_ADDRESS])
633 addr = RTA_DATA (tb[IFA_ADDRESS]);
634 else
635 addr = NULL;
636
637 if (tb[IFA_BROADCAST])
638 broad = RTA_DATA(tb[IFA_BROADCAST]);
639 else
640 broad = NULL;
paul7021c422003-07-15 12:52:22 +0000641 }
paul00df0c12002-12-13 21:07:36 +0000642
paul718e3742002-12-13 20:15:29 +0000643 /* Flags. */
644 if (ifa->ifa_flags & IFA_F_SECONDARY)
645 SET_FLAG (flags, ZEBRA_IFA_SECONDARY);
646
647 /* Label */
648 if (tb[IFA_LABEL])
649 label = (char *) RTA_DATA (tb[IFA_LABEL]);
650
651 if (ifp && label && strcmp (ifp->name, label) == 0)
652 label = NULL;
653
654 /* Register interface address to the interface. */
655 if (ifa->ifa_family == AF_INET)
656 {
paul7021c422003-07-15 12:52:22 +0000657 if (h->nlmsg_type == RTM_NEWADDR)
658 connected_add_ipv4 (ifp, flags,
659 (struct in_addr *) addr, ifa->ifa_prefixlen,
660 (struct in_addr *) broad, label);
661 else
662 connected_delete_ipv4 (ifp, flags,
663 (struct in_addr *) addr, ifa->ifa_prefixlen,
664 (struct in_addr *) broad, label);
paul718e3742002-12-13 20:15:29 +0000665 }
666#ifdef HAVE_IPV6
667 if (ifa->ifa_family == AF_INET6)
668 {
669 if (h->nlmsg_type == RTM_NEWADDR)
paul7021c422003-07-15 12:52:22 +0000670 connected_add_ipv6 (ifp,
671 (struct in6_addr *) addr, ifa->ifa_prefixlen,
672 (struct in6_addr *) broad);
paul718e3742002-12-13 20:15:29 +0000673 else
paul7021c422003-07-15 12:52:22 +0000674 connected_delete_ipv6 (ifp,
675 (struct in6_addr *) addr, ifa->ifa_prefixlen,
676 (struct in6_addr *) broad);
paul718e3742002-12-13 20:15:29 +0000677 }
paul7021c422003-07-15 12:52:22 +0000678#endif /* HAVE_IPV6 */
paul718e3742002-12-13 20:15:29 +0000679
680 return 0;
681}
682
683/* Looking up routing table by netlink interface. */
684int
685netlink_routing_table (struct sockaddr_nl *snl, struct nlmsghdr *h)
686{
687 int len;
688 struct rtmsg *rtm;
paul7021c422003-07-15 12:52:22 +0000689 struct rtattr *tb[RTA_MAX + 1];
paul718e3742002-12-13 20:15:29 +0000690 u_char flags = 0;
paul7021c422003-07-15 12:52:22 +0000691
692 char anyaddr[16] = { 0 };
paul718e3742002-12-13 20:15:29 +0000693
694 int index;
695 int table;
hasso34195bf2004-04-06 12:07:06 +0000696 int metric;
697
paul718e3742002-12-13 20:15:29 +0000698 void *dest;
699 void *gate;
700
701 rtm = NLMSG_DATA (h);
702
703 if (h->nlmsg_type != RTM_NEWROUTE)
704 return 0;
705 if (rtm->rtm_type != RTN_UNICAST)
706 return 0;
707
708 table = rtm->rtm_table;
paul7021c422003-07-15 12:52:22 +0000709#if 0 /* we weed them out later in rib_weed_tables () */
paulb21b19c2003-06-15 01:28:29 +0000710 if (table != RT_TABLE_MAIN && table != zebrad.rtm_table_default)
paul718e3742002-12-13 20:15:29 +0000711 return 0;
712#endif
713
paul7021c422003-07-15 12:52:22 +0000714 len = h->nlmsg_len - NLMSG_LENGTH (sizeof (struct rtmsg));
paul718e3742002-12-13 20:15:29 +0000715 if (len < 0)
716 return -1;
717
718 memset (tb, 0, sizeof tb);
719 netlink_parse_rtattr (tb, RTA_MAX, RTM_RTA (rtm), len);
720
721 if (rtm->rtm_flags & RTM_F_CLONED)
722 return 0;
723 if (rtm->rtm_protocol == RTPROT_REDIRECT)
724 return 0;
725 if (rtm->rtm_protocol == RTPROT_KERNEL)
726 return 0;
727
728 if (rtm->rtm_src_len != 0)
729 return 0;
730
731 /* Route which inserted by Zebra. */
732 if (rtm->rtm_protocol == RTPROT_ZEBRA)
733 flags |= ZEBRA_FLAG_SELFROUTE;
paul7021c422003-07-15 12:52:22 +0000734
paul718e3742002-12-13 20:15:29 +0000735 index = 0;
hasso34195bf2004-04-06 12:07:06 +0000736 metric = 0;
paul718e3742002-12-13 20:15:29 +0000737 dest = NULL;
738 gate = NULL;
739
740 if (tb[RTA_OIF])
741 index = *(int *) RTA_DATA (tb[RTA_OIF]);
742
743 if (tb[RTA_DST])
744 dest = RTA_DATA (tb[RTA_DST]);
745 else
746 dest = anyaddr;
747
748 /* Multipath treatment is needed. */
749 if (tb[RTA_GATEWAY])
750 gate = RTA_DATA (tb[RTA_GATEWAY]);
751
hasso34195bf2004-04-06 12:07:06 +0000752 if (tb[RTA_PRIORITY])
753 metric = *(int *) RTA_DATA(tb[RTA_PRIORITY]);
754
paul718e3742002-12-13 20:15:29 +0000755 if (rtm->rtm_family == AF_INET)
756 {
757 struct prefix_ipv4 p;
758 p.family = AF_INET;
759 memcpy (&p.prefix, dest, 4);
760 p.prefixlen = rtm->rtm_dst_len;
761
hasso34195bf2004-04-06 12:07:06 +0000762 rib_add_ipv4 (ZEBRA_ROUTE_KERNEL, flags, &p, gate, index, table, metric, 0);
paul718e3742002-12-13 20:15:29 +0000763 }
764#ifdef HAVE_IPV6
765 if (rtm->rtm_family == AF_INET6)
766 {
767 struct prefix_ipv6 p;
768 p.family = AF_INET6;
769 memcpy (&p.prefix, dest, 16);
770 p.prefixlen = rtm->rtm_dst_len;
771
772 rib_add_ipv6 (ZEBRA_ROUTE_KERNEL, flags, &p, gate, index, table);
773 }
774#endif /* HAVE_IPV6 */
775
776 return 0;
777}
778
paul7021c422003-07-15 12:52:22 +0000779struct message rtproto_str[] = {
paul718e3742002-12-13 20:15:29 +0000780 {RTPROT_REDIRECT, "redirect"},
781 {RTPROT_KERNEL, "kernel"},
782 {RTPROT_BOOT, "boot"},
783 {RTPROT_STATIC, "static"},
784 {RTPROT_GATED, "GateD"},
785 {RTPROT_RA, "router advertisement"},
786 {RTPROT_MRT, "MRT"},
787 {RTPROT_ZEBRA, "Zebra"},
788#ifdef RTPROT_BIRD
789 {RTPROT_BIRD, "BIRD"},
790#endif /* RTPROT_BIRD */
791 {0, NULL}
792};
793
794/* Routing information change from the kernel. */
795int
796netlink_route_change (struct sockaddr_nl *snl, struct nlmsghdr *h)
797{
798 int len;
799 struct rtmsg *rtm;
paul7021c422003-07-15 12:52:22 +0000800 struct rtattr *tb[RTA_MAX + 1];
801
802 char anyaddr[16] = { 0 };
paul718e3742002-12-13 20:15:29 +0000803
804 int index;
805 int table;
806 void *dest;
807 void *gate;
808
809 rtm = NLMSG_DATA (h);
810
paul7021c422003-07-15 12:52:22 +0000811 if (!(h->nlmsg_type == RTM_NEWROUTE || h->nlmsg_type == RTM_DELROUTE))
paul718e3742002-12-13 20:15:29 +0000812 {
813 /* If this is not route add/delete message print warning. */
814 zlog_warn ("Kernel message: %d\n", h->nlmsg_type);
815 return 0;
816 }
817
818 /* Connected route. */
819 if (IS_ZEBRA_DEBUG_KERNEL)
ajsb6178002004-12-07 21:12:56 +0000820 zlog_debug ("%s %s %s proto %s",
paul7021c422003-07-15 12:52:22 +0000821 h->nlmsg_type ==
822 RTM_NEWROUTE ? "RTM_NEWROUTE" : "RTM_DELROUTE",
823 rtm->rtm_family == AF_INET ? "ipv4" : "ipv6",
824 rtm->rtm_type == RTN_UNICAST ? "unicast" : "multicast",
825 lookup (rtproto_str, rtm->rtm_protocol));
paul718e3742002-12-13 20:15:29 +0000826
827 if (rtm->rtm_type != RTN_UNICAST)
828 {
829 return 0;
830 }
831
832 table = rtm->rtm_table;
paulb21b19c2003-06-15 01:28:29 +0000833 if (table != RT_TABLE_MAIN && table != zebrad.rtm_table_default)
paul718e3742002-12-13 20:15:29 +0000834 {
835 return 0;
836 }
837
paul7021c422003-07-15 12:52:22 +0000838 len = h->nlmsg_len - NLMSG_LENGTH (sizeof (struct rtmsg));
paul718e3742002-12-13 20:15:29 +0000839 if (len < 0)
840 return -1;
841
842 memset (tb, 0, sizeof tb);
843 netlink_parse_rtattr (tb, RTA_MAX, RTM_RTA (rtm), len);
844
845 if (rtm->rtm_flags & RTM_F_CLONED)
846 return 0;
847 if (rtm->rtm_protocol == RTPROT_REDIRECT)
848 return 0;
849 if (rtm->rtm_protocol == RTPROT_KERNEL)
850 return 0;
851
852 if (rtm->rtm_protocol == RTPROT_ZEBRA && h->nlmsg_type == RTM_NEWROUTE)
853 return 0;
854
855 if (rtm->rtm_src_len != 0)
856 {
857 zlog_warn ("netlink_route_change(): no src len");
858 return 0;
859 }
paul7021c422003-07-15 12:52:22 +0000860
paul718e3742002-12-13 20:15:29 +0000861 index = 0;
862 dest = NULL;
863 gate = NULL;
864
865 if (tb[RTA_OIF])
866 index = *(int *) RTA_DATA (tb[RTA_OIF]);
867
868 if (tb[RTA_DST])
869 dest = RTA_DATA (tb[RTA_DST]);
870 else
871 dest = anyaddr;
872
873 if (tb[RTA_GATEWAY])
874 gate = RTA_DATA (tb[RTA_GATEWAY]);
875
876 if (rtm->rtm_family == AF_INET)
877 {
878 struct prefix_ipv4 p;
879 p.family = AF_INET;
880 memcpy (&p.prefix, dest, 4);
881 p.prefixlen = rtm->rtm_dst_len;
882
883 if (IS_ZEBRA_DEBUG_KERNEL)
paul7021c422003-07-15 12:52:22 +0000884 {
885 if (h->nlmsg_type == RTM_NEWROUTE)
ajsb6178002004-12-07 21:12:56 +0000886 zlog_debug ("RTM_NEWROUTE %s/%d",
paul7021c422003-07-15 12:52:22 +0000887 inet_ntoa (p.prefix), p.prefixlen);
888 else
ajsb6178002004-12-07 21:12:56 +0000889 zlog_debug ("RTM_DELROUTE %s/%d",
paul7021c422003-07-15 12:52:22 +0000890 inet_ntoa (p.prefix), p.prefixlen);
891 }
paul718e3742002-12-13 20:15:29 +0000892
893 if (h->nlmsg_type == RTM_NEWROUTE)
paul7021c422003-07-15 12:52:22 +0000894 rib_add_ipv4 (ZEBRA_ROUTE_KERNEL, 0, &p, gate, index, table, 0, 0);
paul718e3742002-12-13 20:15:29 +0000895 else
paul7021c422003-07-15 12:52:22 +0000896 rib_delete_ipv4 (ZEBRA_ROUTE_KERNEL, 0, &p, gate, index, table);
paul718e3742002-12-13 20:15:29 +0000897 }
898
899#ifdef HAVE_IPV6
900 if (rtm->rtm_family == AF_INET6)
901 {
902 struct prefix_ipv6 p;
903 char buf[BUFSIZ];
904
905 p.family = AF_INET6;
906 memcpy (&p.prefix, dest, 16);
907 p.prefixlen = rtm->rtm_dst_len;
908
909 if (IS_ZEBRA_DEBUG_KERNEL)
paul7021c422003-07-15 12:52:22 +0000910 {
911 if (h->nlmsg_type == RTM_NEWROUTE)
ajsb6178002004-12-07 21:12:56 +0000912 zlog_debug ("RTM_NEWROUTE %s/%d",
paul7021c422003-07-15 12:52:22 +0000913 inet_ntop (AF_INET6, &p.prefix, buf, BUFSIZ),
914 p.prefixlen);
915 else
ajsb6178002004-12-07 21:12:56 +0000916 zlog_debug ("RTM_DELROUTE %s/%d",
paul7021c422003-07-15 12:52:22 +0000917 inet_ntop (AF_INET6, &p.prefix, buf, BUFSIZ),
918 p.prefixlen);
919 }
paul718e3742002-12-13 20:15:29 +0000920
921 if (h->nlmsg_type == RTM_NEWROUTE)
paul7021c422003-07-15 12:52:22 +0000922 rib_add_ipv6 (ZEBRA_ROUTE_KERNEL, 0, &p, gate, index, 0);
paul718e3742002-12-13 20:15:29 +0000923 else
paul7021c422003-07-15 12:52:22 +0000924 rib_delete_ipv6 (ZEBRA_ROUTE_KERNEL, 0, &p, gate, index, 0);
paul718e3742002-12-13 20:15:29 +0000925 }
926#endif /* HAVE_IPV6 */
927
928 return 0;
929}
930
931int
932netlink_link_change (struct sockaddr_nl *snl, struct nlmsghdr *h)
933{
934 int len;
935 struct ifinfomsg *ifi;
paul7021c422003-07-15 12:52:22 +0000936 struct rtattr *tb[IFLA_MAX + 1];
paul718e3742002-12-13 20:15:29 +0000937 struct interface *ifp;
938 char *name;
939
940 ifi = NLMSG_DATA (h);
941
paul7021c422003-07-15 12:52:22 +0000942 if (!(h->nlmsg_type == RTM_NEWLINK || h->nlmsg_type == RTM_DELLINK))
paul718e3742002-12-13 20:15:29 +0000943 {
944 /* If this is not link add/delete message so print warning. */
945 zlog_warn ("netlink_link_change: wrong kernel message %d\n",
paul7021c422003-07-15 12:52:22 +0000946 h->nlmsg_type);
paul718e3742002-12-13 20:15:29 +0000947 return 0;
948 }
949
950 len = h->nlmsg_len - NLMSG_LENGTH (sizeof (struct ifinfomsg));
951 if (len < 0)
952 return -1;
953
954 /* Looking up interface name. */
955 memset (tb, 0, sizeof tb);
956 netlink_parse_rtattr (tb, IFLA_MAX, IFLA_RTA (ifi), len);
paulc15cb242005-01-24 09:05:27 +0000957
958 /* check for wireless messages to ignore */
959 if ((tb[IFLA_WIRELESS] != NULL) && (ifi->ifi_change == 0))
960 {
961 if (IS_ZEBRA_DEBUG_KERNEL)
962 zlog_debug ("%s: ignoring IFLA_WIRELESS message", __func__);
963 return 0;
964 }
965
paul718e3742002-12-13 20:15:29 +0000966 if (tb[IFLA_IFNAME] == NULL)
967 return -1;
paul7021c422003-07-15 12:52:22 +0000968 name = (char *) RTA_DATA (tb[IFLA_IFNAME]);
paul718e3742002-12-13 20:15:29 +0000969
970 /* Add interface. */
971 if (h->nlmsg_type == RTM_NEWLINK)
972 {
973 ifp = if_lookup_by_name (name);
974
paul7021c422003-07-15 12:52:22 +0000975 if (ifp == NULL || !CHECK_FLAG (ifp->status, ZEBRA_INTERFACE_ACTIVE))
976 {
977 if (ifp == NULL)
978 ifp = if_get_by_name (name);
paul718e3742002-12-13 20:15:29 +0000979
paul7021c422003-07-15 12:52:22 +0000980 ifp->ifindex = ifi->ifi_index;
981 ifp->flags = ifi->ifi_flags & 0x0000fffff;
paul44145db2004-05-09 11:00:23 +0000982 ifp->mtu6 = ifp->mtu = *(int *) RTA_DATA (tb[IFLA_MTU]);
paul7021c422003-07-15 12:52:22 +0000983 ifp->metric = 1;
paul718e3742002-12-13 20:15:29 +0000984
paul7021c422003-07-15 12:52:22 +0000985 /* If new link is added. */
986 if_add_update (ifp);
987 }
paul718e3742002-12-13 20:15:29 +0000988 else
paul7021c422003-07-15 12:52:22 +0000989 {
990 /* Interface status change. */
991 ifp->ifindex = ifi->ifi_index;
paul44145db2004-05-09 11:00:23 +0000992 ifp->mtu6 = ifp->mtu = *(int *) RTA_DATA (tb[IFLA_MTU]);
paul7021c422003-07-15 12:52:22 +0000993 ifp->metric = 1;
paul718e3742002-12-13 20:15:29 +0000994
paul7021c422003-07-15 12:52:22 +0000995 if (if_is_operative (ifp))
996 {
997 ifp->flags = ifi->ifi_flags & 0x0000fffff;
998 if (!if_is_operative (ifp))
999 if_down (ifp);
1000 }
1001 else
1002 {
1003 ifp->flags = ifi->ifi_flags & 0x0000fffff;
1004 if (if_is_operative (ifp))
1005 if_up (ifp);
1006 }
1007 }
paul718e3742002-12-13 20:15:29 +00001008 }
1009 else
1010 {
1011 /* RTM_DELLINK. */
1012 ifp = if_lookup_by_name (name);
1013
1014 if (ifp == NULL)
paul7021c422003-07-15 12:52:22 +00001015 {
1016 zlog (NULL, LOG_WARNING, "interface %s is deleted but can't find",
paul718e3742002-12-13 20:15:29 +00001017 name);
paul7021c422003-07-15 12:52:22 +00001018 return 0;
1019 }
1020
paul718e3742002-12-13 20:15:29 +00001021 if_delete_update (ifp);
1022 }
1023
1024 return 0;
1025}
1026
1027int
1028netlink_information_fetch (struct sockaddr_nl *snl, struct nlmsghdr *h)
1029{
1030 switch (h->nlmsg_type)
1031 {
1032 case RTM_NEWROUTE:
1033 return netlink_route_change (snl, h);
1034 break;
1035 case RTM_DELROUTE:
1036 return netlink_route_change (snl, h);
1037 break;
1038 case RTM_NEWLINK:
1039 return netlink_link_change (snl, h);
1040 break;
1041 case RTM_DELLINK:
1042 return netlink_link_change (snl, h);
1043 break;
1044 case RTM_NEWADDR:
1045 return netlink_interface_addr (snl, h);
1046 break;
1047 case RTM_DELADDR:
1048 return netlink_interface_addr (snl, h);
1049 break;
1050 default:
1051 zlog_warn ("Unknown netlink nlmsg_type %d\n", h->nlmsg_type);
1052 break;
1053 }
1054 return 0;
1055}
1056
1057/* Interface lookup by netlink socket. */
1058int
1059interface_lookup_netlink ()
1060{
1061 int ret;
paul5f37d862003-04-19 00:11:28 +00001062 int flags;
1063 int snb_ret;
paul7021c422003-07-15 12:52:22 +00001064
paul5f37d862003-04-19 00:11:28 +00001065 /*
1066 * Change netlink socket flags to blocking to ensure we get
1067 * a reply via nelink_parse_info
paul7021c422003-07-15 12:52:22 +00001068 */
1069 snb_ret = set_netlink_blocking (&netlink_cmd, &flags);
1070 if (snb_ret < 0)
1071 zlog (NULL, LOG_WARNING,
1072 "%s:%i Warning: Could not set netlink socket to blocking.",
1073 __FUNCTION__, __LINE__);
1074
paul718e3742002-12-13 20:15:29 +00001075 /* Get interface information. */
1076 ret = netlink_request (AF_PACKET, RTM_GETLINK, &netlink_cmd);
1077 if (ret < 0)
1078 return ret;
1079 ret = netlink_parse_info (netlink_interface, &netlink_cmd);
1080 if (ret < 0)
1081 return ret;
1082
1083 /* Get IPv4 address of the interfaces. */
1084 ret = netlink_request (AF_INET, RTM_GETADDR, &netlink_cmd);
1085 if (ret < 0)
1086 return ret;
1087 ret = netlink_parse_info (netlink_interface_addr, &netlink_cmd);
1088 if (ret < 0)
1089 return ret;
1090
1091#ifdef HAVE_IPV6
1092 /* Get IPv6 address of the interfaces. */
1093 ret = netlink_request (AF_INET6, RTM_GETADDR, &netlink_cmd);
1094 if (ret < 0)
1095 return ret;
1096 ret = netlink_parse_info (netlink_interface_addr, &netlink_cmd);
1097 if (ret < 0)
1098 return ret;
1099#endif /* HAVE_IPV6 */
1100
paul7021c422003-07-15 12:52:22 +00001101 /* restore socket flags */
1102 if (snb_ret == 0)
1103 set_netlink_nonblocking (&netlink_cmd, &flags);
paul718e3742002-12-13 20:15:29 +00001104 return 0;
1105}
1106
1107/* Routing table read function using netlink interface. Only called
1108 bootstrap time. */
1109int
1110netlink_route_read ()
1111{
1112 int ret;
paul5f37d862003-04-19 00:11:28 +00001113 int flags;
1114 int snb_ret;
paul7021c422003-07-15 12:52:22 +00001115
paul5f37d862003-04-19 00:11:28 +00001116 /*
1117 * Change netlink socket flags to blocking to ensure we get
1118 * a reply via nelink_parse_info
paul7021c422003-07-15 12:52:22 +00001119 */
1120 snb_ret = set_netlink_blocking (&netlink_cmd, &flags);
1121 if (snb_ret < 0)
1122 zlog (NULL, LOG_WARNING,
1123 "%s:%i Warning: Could not set netlink socket to blocking.",
1124 __FUNCTION__, __LINE__);
1125
paul718e3742002-12-13 20:15:29 +00001126 /* Get IPv4 routing table. */
1127 ret = netlink_request (AF_INET, RTM_GETROUTE, &netlink_cmd);
1128 if (ret < 0)
1129 return ret;
1130 ret = netlink_parse_info (netlink_routing_table, &netlink_cmd);
1131 if (ret < 0)
1132 return ret;
1133
1134#ifdef HAVE_IPV6
1135 /* Get IPv6 routing table. */
1136 ret = netlink_request (AF_INET6, RTM_GETROUTE, &netlink_cmd);
1137 if (ret < 0)
1138 return ret;
1139 ret = netlink_parse_info (netlink_routing_table, &netlink_cmd);
1140 if (ret < 0)
1141 return ret;
1142#endif /* HAVE_IPV6 */
1143
paul5f37d862003-04-19 00:11:28 +00001144 /* restore flags */
paul7021c422003-07-15 12:52:22 +00001145 if (snb_ret == 0)
1146 set_netlink_nonblocking (&netlink_cmd, &flags);
paul718e3742002-12-13 20:15:29 +00001147 return 0;
1148}
1149
1150/* Utility function comes from iproute2.
1151 Authors: Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru> */
1152int
1153addattr_l (struct nlmsghdr *n, int maxlen, int type, void *data, int alen)
1154{
1155 int len;
1156 struct rtattr *rta;
1157
paul7021c422003-07-15 12:52:22 +00001158 len = RTA_LENGTH (alen);
paul718e3742002-12-13 20:15:29 +00001159
paul7021c422003-07-15 12:52:22 +00001160 if (NLMSG_ALIGN (n->nlmsg_len) + len > maxlen)
paul718e3742002-12-13 20:15:29 +00001161 return -1;
1162
paul7021c422003-07-15 12:52:22 +00001163 rta = (struct rtattr *) (((char *) n) + NLMSG_ALIGN (n->nlmsg_len));
paul718e3742002-12-13 20:15:29 +00001164 rta->rta_type = type;
1165 rta->rta_len = len;
paul7021c422003-07-15 12:52:22 +00001166 memcpy (RTA_DATA (rta), data, alen);
paul718e3742002-12-13 20:15:29 +00001167 n->nlmsg_len = NLMSG_ALIGN (n->nlmsg_len) + len;
1168
1169 return 0;
1170}
1171
1172int
1173rta_addattr_l (struct rtattr *rta, int maxlen, int type, void *data, int alen)
1174{
1175 int len;
1176 struct rtattr *subrta;
1177
paul7021c422003-07-15 12:52:22 +00001178 len = RTA_LENGTH (alen);
paul718e3742002-12-13 20:15:29 +00001179
paul7021c422003-07-15 12:52:22 +00001180 if (RTA_ALIGN (rta->rta_len) + len > maxlen)
paul718e3742002-12-13 20:15:29 +00001181 return -1;
1182
paul7021c422003-07-15 12:52:22 +00001183 subrta = (struct rtattr *) (((char *) rta) + RTA_ALIGN (rta->rta_len));
paul718e3742002-12-13 20:15:29 +00001184 subrta->rta_type = type;
1185 subrta->rta_len = len;
paul7021c422003-07-15 12:52:22 +00001186 memcpy (RTA_DATA (subrta), data, alen);
paul718e3742002-12-13 20:15:29 +00001187 rta->rta_len = NLMSG_ALIGN (rta->rta_len) + len;
1188
1189 return 0;
1190}
1191
1192/* Utility function comes from iproute2.
1193 Authors: Alexey Kuznetsov, <kuznet@ms2.inr.ac.ru> */
1194int
1195addattr32 (struct nlmsghdr *n, int maxlen, int type, int data)
1196{
1197 int len;
1198 struct rtattr *rta;
paul7021c422003-07-15 12:52:22 +00001199
1200 len = RTA_LENGTH (4);
1201
paul718e3742002-12-13 20:15:29 +00001202 if (NLMSG_ALIGN (n->nlmsg_len) + len > maxlen)
1203 return -1;
1204
paul7021c422003-07-15 12:52:22 +00001205 rta = (struct rtattr *) (((char *) n) + NLMSG_ALIGN (n->nlmsg_len));
paul718e3742002-12-13 20:15:29 +00001206 rta->rta_type = type;
1207 rta->rta_len = len;
paul7021c422003-07-15 12:52:22 +00001208 memcpy (RTA_DATA (rta), &data, 4);
paul718e3742002-12-13 20:15:29 +00001209 n->nlmsg_len = NLMSG_ALIGN (n->nlmsg_len) + len;
1210
1211 return 0;
1212}
1213
1214static int
1215netlink_talk_filter (struct sockaddr_nl *snl, struct nlmsghdr *h)
1216{
1217 zlog_warn ("netlink_talk: ignoring message type 0x%04x", h->nlmsg_type);
1218 return 0;
1219}
1220
1221/* sendmsg() to netlink socket then recvmsg(). */
1222int
1223netlink_talk (struct nlmsghdr *n, struct nlsock *nl)
1224{
1225 int status;
1226 struct sockaddr_nl snl;
paul7021c422003-07-15 12:52:22 +00001227 struct iovec iov = { (void *) n, n->nlmsg_len };
1228 struct msghdr msg = { (void *) &snl, sizeof snl, &iov, 1, NULL, 0, 0 };
paul718e3742002-12-13 20:15:29 +00001229 int flags = 0;
paul5f37d862003-04-19 00:11:28 +00001230 int snb_ret;
ajs4be019d2005-01-29 16:12:41 +00001231 int save_errno;
paul7021c422003-07-15 12:52:22 +00001232
paul718e3742002-12-13 20:15:29 +00001233 memset (&snl, 0, sizeof snl);
1234 snl.nl_family = AF_NETLINK;
paul7021c422003-07-15 12:52:22 +00001235
paul718e3742002-12-13 20:15:29 +00001236 n->nlmsg_seq = ++netlink_cmd.seq;
1237
1238 /* Request an acknowledgement by setting NLM_F_ACK */
1239 n->nlmsg_flags |= NLM_F_ACK;
paul7021c422003-07-15 12:52:22 +00001240
1241 if (IS_ZEBRA_DEBUG_KERNEL)
ajsb6178002004-12-07 21:12:56 +00001242 zlog_debug ("netlink_talk: %s type %s(%u), seq=%u", netlink_cmd.name,
paul7021c422003-07-15 12:52:22 +00001243 lookup (nlmsg_str, n->nlmsg_type), n->nlmsg_type,
1244 n->nlmsg_seq);
paul718e3742002-12-13 20:15:29 +00001245
1246 /* Send message to netlink interface. */
paul7021c422003-07-15 12:52:22 +00001247 if (zserv_privs.change (ZPRIVS_RAISE))
1248 zlog (NULL, LOG_ERR, "Can't raise privileges");
paul718e3742002-12-13 20:15:29 +00001249 status = sendmsg (nl->sock, &msg, 0);
ajs4be019d2005-01-29 16:12:41 +00001250 save_errno = errno;
paul7021c422003-07-15 12:52:22 +00001251 if (zserv_privs.change (ZPRIVS_LOWER))
1252 zlog (NULL, LOG_ERR, "Can't lower privileges");
1253
paul718e3742002-12-13 20:15:29 +00001254 if (status < 0)
1255 {
1256 zlog (NULL, LOG_ERR, "netlink_talk sendmsg() error: %s",
ajs4be019d2005-01-29 16:12:41 +00001257 safe_strerror (save_errno));
paul718e3742002-12-13 20:15:29 +00001258 return -1;
1259 }
paul7021c422003-07-15 12:52:22 +00001260
paul718e3742002-12-13 20:15:29 +00001261 /*
1262 * Change socket flags for blocking I/O.
1263 * This ensures we wait for a reply in netlink_parse_info().
1264 */
paul7021c422003-07-15 12:52:22 +00001265 snb_ret = set_netlink_blocking (nl, &flags);
1266 if (snb_ret < 0)
1267 zlog (NULL, LOG_WARNING,
1268 "%s:%i Warning: Could not set netlink socket to blocking.",
1269 __FUNCTION__, __LINE__);
paul718e3742002-12-13 20:15:29 +00001270
1271 /*
1272 * Get reply from netlink socket.
1273 * The reply should either be an acknowlegement or an error.
1274 */
1275 status = netlink_parse_info (netlink_talk_filter, nl);
paul7021c422003-07-15 12:52:22 +00001276
paul718e3742002-12-13 20:15:29 +00001277 /* Restore socket flags for nonblocking I/O */
paul7021c422003-07-15 12:52:22 +00001278 if (snb_ret == 0)
1279 set_netlink_nonblocking (nl, &flags);
1280
paul718e3742002-12-13 20:15:29 +00001281 return status;
1282}
1283
1284/* Routing table change via netlink interface. */
1285int
1286netlink_route (int cmd, int family, void *dest, int length, void *gate,
paul7021c422003-07-15 12:52:22 +00001287 int index, int zebra_flags, int table)
paul718e3742002-12-13 20:15:29 +00001288{
1289 int ret;
1290 int bytelen;
1291 struct sockaddr_nl snl;
1292 int discard;
1293
paul7021c422003-07-15 12:52:22 +00001294 struct
paul718e3742002-12-13 20:15:29 +00001295 {
1296 struct nlmsghdr n;
1297 struct rtmsg r;
1298 char buf[1024];
1299 } req;
1300
1301 memset (&req, 0, sizeof req);
1302
1303 bytelen = (family == AF_INET ? 4 : 16);
1304
1305 req.n.nlmsg_len = NLMSG_LENGTH (sizeof (struct rtmsg));
1306 req.n.nlmsg_flags = NLM_F_CREATE | NLM_F_REQUEST;
1307 req.n.nlmsg_type = cmd;
1308 req.r.rtm_family = family;
1309 req.r.rtm_table = table;
1310 req.r.rtm_dst_len = length;
1311
hasso81dfcaa2003-05-25 19:21:25 +00001312 if ((zebra_flags & ZEBRA_FLAG_BLACKHOLE)
1313 || (zebra_flags & ZEBRA_FLAG_REJECT))
paul718e3742002-12-13 20:15:29 +00001314 discard = 1;
1315 else
1316 discard = 0;
1317
paul7021c422003-07-15 12:52:22 +00001318 if (cmd == RTM_NEWROUTE)
paul718e3742002-12-13 20:15:29 +00001319 {
1320 req.r.rtm_protocol = RTPROT_ZEBRA;
1321 req.r.rtm_scope = RT_SCOPE_UNIVERSE;
1322
paul7021c422003-07-15 12:52:22 +00001323 if (discard)
paul595db7f2003-05-25 21:35:06 +00001324 {
1325 if (zebra_flags & ZEBRA_FLAG_BLACKHOLE)
1326 req.r.rtm_type = RTN_BLACKHOLE;
1327 else if (zebra_flags & ZEBRA_FLAG_REJECT)
1328 req.r.rtm_type = RTN_UNREACHABLE;
paul7021c422003-07-15 12:52:22 +00001329 else
1330 assert (RTN_BLACKHOLE != RTN_UNREACHABLE); /* false */
1331 }
paul595db7f2003-05-25 21:35:06 +00001332 else
paul7021c422003-07-15 12:52:22 +00001333 req.r.rtm_type = RTN_UNICAST;
paul718e3742002-12-13 20:15:29 +00001334 }
1335
1336 if (dest)
1337 addattr_l (&req.n, sizeof req, RTA_DST, dest, bytelen);
1338
paul7021c422003-07-15 12:52:22 +00001339 if (!discard)
paul718e3742002-12-13 20:15:29 +00001340 {
1341 if (gate)
paul7021c422003-07-15 12:52:22 +00001342 addattr_l (&req.n, sizeof req, RTA_GATEWAY, gate, bytelen);
paul718e3742002-12-13 20:15:29 +00001343 if (index > 0)
paul7021c422003-07-15 12:52:22 +00001344 addattr32 (&req.n, sizeof req, RTA_OIF, index);
paul718e3742002-12-13 20:15:29 +00001345 }
1346
1347 /* Destination netlink address. */
1348 memset (&snl, 0, sizeof snl);
1349 snl.nl_family = AF_NETLINK;
1350
1351 /* Talk to netlink socket. */
1352 ret = netlink_talk (&req.n, &netlink);
1353 if (ret < 0)
1354 return -1;
1355
1356 return 0;
1357}
1358
1359/* Routing table change via netlink interface. */
1360int
1361netlink_route_multipath (int cmd, struct prefix *p, struct rib *rib,
paul7021c422003-07-15 12:52:22 +00001362 int family)
paul718e3742002-12-13 20:15:29 +00001363{
1364 int bytelen;
1365 struct sockaddr_nl snl;
1366 struct nexthop *nexthop = NULL;
1367 int nexthop_num = 0;
1368 struct nlsock *nl;
1369 int discard;
1370
paul7021c422003-07-15 12:52:22 +00001371 struct
paul718e3742002-12-13 20:15:29 +00001372 {
1373 struct nlmsghdr n;
1374 struct rtmsg r;
1375 char buf[1024];
1376 } req;
1377
1378 memset (&req, 0, sizeof req);
1379
1380 bytelen = (family == AF_INET ? 4 : 16);
1381
1382 req.n.nlmsg_len = NLMSG_LENGTH (sizeof (struct rtmsg));
1383 req.n.nlmsg_flags = NLM_F_CREATE | NLM_F_REQUEST;
1384 req.n.nlmsg_type = cmd;
1385 req.r.rtm_family = family;
1386 req.r.rtm_table = rib->table;
1387 req.r.rtm_dst_len = p->prefixlen;
1388
paul13766da2003-02-07 14:46:23 +00001389#ifdef RTM_F_EQUALIZE
1390 req.r.rtm_flags |= RTM_F_EQUALIZE;
1391#endif /* RTM_F_EQUALIZE */
1392
paul7021c422003-07-15 12:52:22 +00001393 if ((rib->flags & ZEBRA_FLAG_BLACKHOLE) || (rib->flags & ZEBRA_FLAG_REJECT))
paul718e3742002-12-13 20:15:29 +00001394 discard = 1;
1395 else
1396 discard = 0;
1397
paul7021c422003-07-15 12:52:22 +00001398 if (cmd == RTM_NEWROUTE)
paul718e3742002-12-13 20:15:29 +00001399 {
1400 req.r.rtm_protocol = RTPROT_ZEBRA;
1401 req.r.rtm_scope = RT_SCOPE_UNIVERSE;
1402
paul7021c422003-07-15 12:52:22 +00001403 if (discard)
paul595db7f2003-05-25 21:35:06 +00001404 {
1405 if (rib->flags & ZEBRA_FLAG_BLACKHOLE)
1406 req.r.rtm_type = RTN_BLACKHOLE;
1407 else if (rib->flags & ZEBRA_FLAG_REJECT)
1408 req.r.rtm_type = RTN_UNREACHABLE;
paul7021c422003-07-15 12:52:22 +00001409 else
1410 assert (RTN_BLACKHOLE != RTN_UNREACHABLE); /* false */
1411 }
paul595db7f2003-05-25 21:35:06 +00001412 else
paul7021c422003-07-15 12:52:22 +00001413 req.r.rtm_type = RTN_UNICAST;
paul718e3742002-12-13 20:15:29 +00001414 }
1415
1416 addattr_l (&req.n, sizeof req, RTA_DST, &p->u.prefix, bytelen);
1417
1418 /* Metric. */
1419 addattr32 (&req.n, sizeof req, RTA_PRIORITY, rib->metric);
1420
1421 if (discard)
1422 {
1423 if (cmd == RTM_NEWROUTE)
paul7021c422003-07-15 12:52:22 +00001424 for (nexthop = rib->nexthop; nexthop; nexthop = nexthop->next)
1425 SET_FLAG (nexthop->flags, NEXTHOP_FLAG_FIB);
paul718e3742002-12-13 20:15:29 +00001426 goto skip;
1427 }
1428
1429 /* Multipath case. */
1430 if (rib->nexthop_active_num == 1 || MULTIPATH_NUM == 1)
1431 {
1432 for (nexthop = rib->nexthop; nexthop; nexthop = nexthop->next)
paul7021c422003-07-15 12:52:22 +00001433 {
paul5ec90d22003-06-19 01:41:37 +00001434
paul7021c422003-07-15 12:52:22 +00001435 if ((cmd == RTM_NEWROUTE
1436 && CHECK_FLAG (nexthop->flags, NEXTHOP_FLAG_ACTIVE))
1437 || (cmd == RTM_DELROUTE
1438 && CHECK_FLAG (nexthop->flags, NEXTHOP_FLAG_FIB)))
1439 {
paul5ec90d22003-06-19 01:41:37 +00001440
paul7021c422003-07-15 12:52:22 +00001441 if (CHECK_FLAG (nexthop->flags, NEXTHOP_FLAG_RECURSIVE))
1442 {
1443 if (IS_ZEBRA_DEBUG_KERNEL)
1444 {
ajsb6178002004-12-07 21:12:56 +00001445 zlog_debug
paul7021c422003-07-15 12:52:22 +00001446 ("netlink_route_multipath() (recursive, 1 hop): "
1447 "%s %s/%d via %s if %u, type %s",
1448 lookup (nlmsg_str, cmd), inet_ntoa (p->u.prefix4),
1449 p->prefixlen, inet_ntoa (nexthop->rgate.ipv4),
1450 nexthop->rifindex,
1451 nexthop_types_desc[nexthop->rtype]);
1452 }
paul5ec90d22003-06-19 01:41:37 +00001453
paul7021c422003-07-15 12:52:22 +00001454 if (nexthop->rtype == NEXTHOP_TYPE_IPV4
1455 || nexthop->rtype == NEXTHOP_TYPE_IPV4_IFINDEX)
1456 addattr_l (&req.n, sizeof req, RTA_GATEWAY,
1457 &nexthop->rgate.ipv4, bytelen);
paul718e3742002-12-13 20:15:29 +00001458#ifdef HAVE_IPV6
paul7021c422003-07-15 12:52:22 +00001459 if (nexthop->rtype == NEXTHOP_TYPE_IPV6
1460 || nexthop->rtype == NEXTHOP_TYPE_IPV6_IFINDEX
1461 || nexthop->rtype == NEXTHOP_TYPE_IPV6_IFNAME)
1462 addattr_l (&req.n, sizeof req, RTA_GATEWAY,
1463 &nexthop->rgate.ipv6, bytelen);
paul718e3742002-12-13 20:15:29 +00001464#endif /* HAVE_IPV6 */
paul7021c422003-07-15 12:52:22 +00001465 if (nexthop->rtype == NEXTHOP_TYPE_IFINDEX
1466 || nexthop->rtype == NEXTHOP_TYPE_IFNAME
1467 || nexthop->rtype == NEXTHOP_TYPE_IPV4_IFINDEX
1468 || nexthop->rtype == NEXTHOP_TYPE_IPV6_IFINDEX
1469 || nexthop->rtype == NEXTHOP_TYPE_IPV6_IFNAME)
1470 addattr32 (&req.n, sizeof req, RTA_OIF,
1471 nexthop->rifindex);
1472 }
1473 else
1474 {
1475 if (IS_ZEBRA_DEBUG_KERNEL)
1476 {
ajsb6178002004-12-07 21:12:56 +00001477 zlog_debug
paul7021c422003-07-15 12:52:22 +00001478 ("netlink_route_multipath(): (single hop)"
1479 "%s %s/%d via %s if %u, type %s",
1480 lookup (nlmsg_str, cmd), inet_ntoa (p->u.prefix4),
1481 p->prefixlen, inet_ntoa (nexthop->gate.ipv4),
1482 nexthop->ifindex,
1483 nexthop_types_desc[nexthop->type]);
1484 }
paul5ec90d22003-06-19 01:41:37 +00001485
paul7021c422003-07-15 12:52:22 +00001486 if (nexthop->type == NEXTHOP_TYPE_IPV4
1487 || nexthop->type == NEXTHOP_TYPE_IPV4_IFINDEX)
1488 addattr_l (&req.n, sizeof req, RTA_GATEWAY,
1489 &nexthop->gate.ipv4, bytelen);
paul718e3742002-12-13 20:15:29 +00001490#ifdef HAVE_IPV6
paul7021c422003-07-15 12:52:22 +00001491 if (nexthop->type == NEXTHOP_TYPE_IPV6
1492 || nexthop->type == NEXTHOP_TYPE_IPV6_IFNAME
1493 || nexthop->type == NEXTHOP_TYPE_IPV6_IFINDEX)
1494 addattr_l (&req.n, sizeof req, RTA_GATEWAY,
1495 &nexthop->gate.ipv6, bytelen);
paul718e3742002-12-13 20:15:29 +00001496#endif /* HAVE_IPV6 */
paul7021c422003-07-15 12:52:22 +00001497 if (nexthop->type == NEXTHOP_TYPE_IFINDEX
1498 || nexthop->type == NEXTHOP_TYPE_IFNAME
1499 || nexthop->type == NEXTHOP_TYPE_IPV4_IFINDEX
1500 || nexthop->type == NEXTHOP_TYPE_IPV6_IFINDEX
1501 || nexthop->type == NEXTHOP_TYPE_IPV6_IFNAME)
1502 addattr32 (&req.n, sizeof req, RTA_OIF, nexthop->ifindex);
1503 }
paul718e3742002-12-13 20:15:29 +00001504
paul7021c422003-07-15 12:52:22 +00001505 if (cmd == RTM_NEWROUTE)
1506 SET_FLAG (nexthop->flags, NEXTHOP_FLAG_FIB);
paul718e3742002-12-13 20:15:29 +00001507
paul7021c422003-07-15 12:52:22 +00001508 nexthop_num++;
1509 break;
1510 }
1511 }
paul718e3742002-12-13 20:15:29 +00001512 }
1513 else
1514 {
1515 char buf[1024];
1516 struct rtattr *rta = (void *) buf;
1517 struct rtnexthop *rtnh;
1518
1519 rta->rta_type = RTA_MULTIPATH;
paul7021c422003-07-15 12:52:22 +00001520 rta->rta_len = RTA_LENGTH (0);
1521 rtnh = RTA_DATA (rta);
paul718e3742002-12-13 20:15:29 +00001522
1523 nexthop_num = 0;
1524 for (nexthop = rib->nexthop;
paul7021c422003-07-15 12:52:22 +00001525 nexthop && (MULTIPATH_NUM == 0 || nexthop_num < MULTIPATH_NUM);
1526 nexthop = nexthop->next)
1527 {
1528 if ((cmd == RTM_NEWROUTE
1529 && CHECK_FLAG (nexthop->flags, NEXTHOP_FLAG_ACTIVE))
1530 || (cmd == RTM_DELROUTE
1531 && CHECK_FLAG (nexthop->flags, NEXTHOP_FLAG_FIB)))
1532 {
1533 nexthop_num++;
paul718e3742002-12-13 20:15:29 +00001534
paul7021c422003-07-15 12:52:22 +00001535 rtnh->rtnh_len = sizeof (*rtnh);
1536 rtnh->rtnh_flags = 0;
1537 rtnh->rtnh_hops = 0;
1538 rta->rta_len += rtnh->rtnh_len;
paul718e3742002-12-13 20:15:29 +00001539
paul7021c422003-07-15 12:52:22 +00001540 if (CHECK_FLAG (nexthop->flags, NEXTHOP_FLAG_RECURSIVE))
1541 {
1542 if (IS_ZEBRA_DEBUG_KERNEL)
1543 {
ajsb6178002004-12-07 21:12:56 +00001544 zlog_debug ("netlink_route_multipath() "
paul7021c422003-07-15 12:52:22 +00001545 "(recursive, multihop): "
1546 "%s %s/%d via %s if %u, type %s",
1547 lookup (nlmsg_str, cmd), inet_ntoa (p->u.prefix4),
1548 p->prefixlen, inet_ntoa (nexthop->rgate.ipv4),
1549 nexthop->rifindex,
1550 nexthop_types_desc[nexthop->type]);
1551 }
1552 if (nexthop->rtype == NEXTHOP_TYPE_IPV4
1553 || nexthop->rtype == NEXTHOP_TYPE_IPV4_IFINDEX)
1554 {
1555 rta_addattr_l (rta, 4096, RTA_GATEWAY,
1556 &nexthop->rgate.ipv4, bytelen);
1557 rtnh->rtnh_len += sizeof (struct rtattr) + 4;
1558 }
paul718e3742002-12-13 20:15:29 +00001559#ifdef HAVE_IPV6
paul7021c422003-07-15 12:52:22 +00001560 if (nexthop->rtype == NEXTHOP_TYPE_IPV6
1561 || nexthop->rtype == NEXTHOP_TYPE_IPV6_IFNAME
1562 || nexthop->rtype == NEXTHOP_TYPE_IPV6_IFINDEX)
1563 rta_addattr_l (rta, 4096, RTA_GATEWAY,
1564 &nexthop->rgate.ipv6, bytelen);
paul718e3742002-12-13 20:15:29 +00001565#endif /* HAVE_IPV6 */
paul7021c422003-07-15 12:52:22 +00001566 /* ifindex */
1567 if (nexthop->rtype == NEXTHOP_TYPE_IFINDEX
1568 || nexthop->rtype == NEXTHOP_TYPE_IFNAME
1569 || nexthop->rtype == NEXTHOP_TYPE_IPV4_IFINDEX
1570 || nexthop->rtype == NEXTHOP_TYPE_IPV6_IFINDEX
1571 || nexthop->rtype == NEXTHOP_TYPE_IPV6_IFNAME)
1572 rtnh->rtnh_ifindex = nexthop->rifindex;
1573 else
1574 rtnh->rtnh_ifindex = 0;
1575 }
1576 else
1577 {
1578 if (IS_ZEBRA_DEBUG_KERNEL)
1579 {
ajsb6178002004-12-07 21:12:56 +00001580 zlog_debug ("netlink_route_multipath() "
paul7021c422003-07-15 12:52:22 +00001581 "(multihop): "
1582 "%s %s/%d via %s if %u, type %s",
1583 lookup (nlmsg_str, cmd), inet_ntoa (p->u.prefix4),
1584 p->prefixlen, inet_ntoa (nexthop->rgate.ipv4),
1585 nexthop->rifindex,
1586 nexthop_types_desc[nexthop->type]);
1587 }
1588 if (nexthop->type == NEXTHOP_TYPE_IPV4
1589 || nexthop->type == NEXTHOP_TYPE_IPV4_IFINDEX)
1590 {
1591 rta_addattr_l (rta, 4096, RTA_GATEWAY,
1592 &nexthop->gate.ipv4, bytelen);
1593 rtnh->rtnh_len += sizeof (struct rtattr) + 4;
1594 }
paul718e3742002-12-13 20:15:29 +00001595#ifdef HAVE_IPV6
paul7021c422003-07-15 12:52:22 +00001596 if (nexthop->type == NEXTHOP_TYPE_IPV6
1597 || nexthop->type == NEXTHOP_TYPE_IPV6_IFNAME
1598 || nexthop->type == NEXTHOP_TYPE_IPV6_IFINDEX)
1599 rta_addattr_l (rta, 4096, RTA_GATEWAY,
1600 &nexthop->gate.ipv6, bytelen);
paul718e3742002-12-13 20:15:29 +00001601#endif /* HAVE_IPV6 */
paul7021c422003-07-15 12:52:22 +00001602 /* ifindex */
1603 if (nexthop->type == NEXTHOP_TYPE_IFINDEX
1604 || nexthop->type == NEXTHOP_TYPE_IFNAME
1605 || nexthop->type == NEXTHOP_TYPE_IPV4_IFINDEX
1606 || nexthop->type == NEXTHOP_TYPE_IPV6_IFNAME
1607 || nexthop->type == NEXTHOP_TYPE_IPV6_IFINDEX)
1608 rtnh->rtnh_ifindex = nexthop->ifindex;
1609 else
1610 rtnh->rtnh_ifindex = 0;
1611 }
1612 rtnh = RTNH_NEXT (rtnh);
paul718e3742002-12-13 20:15:29 +00001613
paul7021c422003-07-15 12:52:22 +00001614 if (cmd == RTM_NEWROUTE)
1615 SET_FLAG (nexthop->flags, NEXTHOP_FLAG_FIB);
1616 }
1617 }
paul718e3742002-12-13 20:15:29 +00001618
1619 if (rta->rta_len > RTA_LENGTH (0))
paul7021c422003-07-15 12:52:22 +00001620 addattr_l (&req.n, 1024, RTA_MULTIPATH, RTA_DATA (rta),
1621 RTA_PAYLOAD (rta));
paul718e3742002-12-13 20:15:29 +00001622 }
1623
1624 /* If there is no useful nexthop then return. */
1625 if (nexthop_num == 0)
1626 {
1627 if (IS_ZEBRA_DEBUG_KERNEL)
ajsb6178002004-12-07 21:12:56 +00001628 zlog_debug ("netlink_route_multipath(): No useful nexthop.");
paul718e3742002-12-13 20:15:29 +00001629 return 0;
1630 }
1631
paul7021c422003-07-15 12:52:22 +00001632skip:
paul718e3742002-12-13 20:15:29 +00001633
1634 /* Destination netlink address. */
1635 memset (&snl, 0, sizeof snl);
1636 snl.nl_family = AF_NETLINK;
1637
1638 if (family == AF_INET)
1639 nl = &netlink_cmd;
1640 else
1641 nl = &netlink;
1642
1643 /* Talk to netlink socket. */
1644 return netlink_talk (&req.n, nl);
1645}
1646
1647int
1648kernel_add_ipv4 (struct prefix *p, struct rib *rib)
1649{
1650 return netlink_route_multipath (RTM_NEWROUTE, p, rib, AF_INET);
1651}
1652
1653int
1654kernel_delete_ipv4 (struct prefix *p, struct rib *rib)
1655{
1656 return netlink_route_multipath (RTM_DELROUTE, p, rib, AF_INET);
1657}
1658
1659#ifdef HAVE_IPV6
1660int
1661kernel_add_ipv6 (struct prefix *p, struct rib *rib)
1662{
1663 return netlink_route_multipath (RTM_NEWROUTE, p, rib, AF_INET6);
1664}
1665
1666int
1667kernel_delete_ipv6 (struct prefix *p, struct rib *rib)
1668{
1669 return netlink_route_multipath (RTM_DELROUTE, p, rib, AF_INET6);
1670}
1671
1672/* Delete IPv6 route from the kernel. */
1673int
1674kernel_delete_ipv6_old (struct prefix_ipv6 *dest, struct in6_addr *gate,
paul7021c422003-07-15 12:52:22 +00001675 int index, int flags, int table)
paul718e3742002-12-13 20:15:29 +00001676{
paul7021c422003-07-15 12:52:22 +00001677 return netlink_route (RTM_DELROUTE, AF_INET6, &dest->prefix,
1678 dest->prefixlen, gate, index, flags, table);
paul718e3742002-12-13 20:15:29 +00001679}
1680#endif /* HAVE_IPV6 */
1681
1682/* Interface address modification. */
1683int
1684netlink_address (int cmd, int family, struct interface *ifp,
paul7021c422003-07-15 12:52:22 +00001685 struct connected *ifc)
paul718e3742002-12-13 20:15:29 +00001686{
1687 int bytelen;
1688 struct prefix *p;
1689
paul7021c422003-07-15 12:52:22 +00001690 struct
paul718e3742002-12-13 20:15:29 +00001691 {
1692 struct nlmsghdr n;
1693 struct ifaddrmsg ifa;
1694 char buf[1024];
1695 } req;
1696
1697 p = ifc->address;
1698 memset (&req, 0, sizeof req);
1699
1700 bytelen = (family == AF_INET ? 4 : 16);
1701
paul7021c422003-07-15 12:52:22 +00001702 req.n.nlmsg_len = NLMSG_LENGTH (sizeof (struct ifaddrmsg));
paul718e3742002-12-13 20:15:29 +00001703 req.n.nlmsg_flags = NLM_F_REQUEST;
1704 req.n.nlmsg_type = cmd;
1705 req.ifa.ifa_family = family;
1706
1707 req.ifa.ifa_index = ifp->ifindex;
1708 req.ifa.ifa_prefixlen = p->prefixlen;
1709
1710 addattr_l (&req.n, sizeof req, IFA_LOCAL, &p->u.prefix, bytelen);
1711
1712 if (family == AF_INET && cmd == RTM_NEWADDR)
1713 {
1714 if (if_is_broadcast (ifp) && ifc->destination)
paul7021c422003-07-15 12:52:22 +00001715 {
1716 p = ifc->destination;
1717 addattr_l (&req.n, sizeof req, IFA_BROADCAST, &p->u.prefix,
1718 bytelen);
1719 }
paul718e3742002-12-13 20:15:29 +00001720 }
1721
1722 if (CHECK_FLAG (ifc->flags, ZEBRA_IFA_SECONDARY))
1723 SET_FLAG (req.ifa.ifa_flags, IFA_F_SECONDARY);
paul7021c422003-07-15 12:52:22 +00001724
paul718e3742002-12-13 20:15:29 +00001725 if (ifc->label)
1726 addattr_l (&req.n, sizeof req, IFA_LABEL, ifc->label,
paul7021c422003-07-15 12:52:22 +00001727 strlen (ifc->label) + 1);
paul718e3742002-12-13 20:15:29 +00001728
1729 return netlink_talk (&req.n, &netlink_cmd);
1730}
1731
1732int
1733kernel_address_add_ipv4 (struct interface *ifp, struct connected *ifc)
1734{
1735 return netlink_address (RTM_NEWADDR, AF_INET, ifp, ifc);
1736}
1737
1738int
1739kernel_address_delete_ipv4 (struct interface *ifp, struct connected *ifc)
1740{
1741 return netlink_address (RTM_DELADDR, AF_INET, ifp, ifc);
1742}
1743
paul718e3742002-12-13 20:15:29 +00001744
1745extern struct thread_master *master;
1746
1747/* Kernel route reflection. */
1748int
1749kernel_read (struct thread *thread)
1750{
1751 int ret;
1752 int sock;
1753
1754 sock = THREAD_FD (thread);
1755 ret = netlink_parse_info (netlink_information_fetch, &netlink);
paulb21b19c2003-06-15 01:28:29 +00001756 thread_add_read (zebrad.master, kernel_read, NULL, netlink.sock);
paul718e3742002-12-13 20:15:29 +00001757
1758 return 0;
1759}
1760
1761/* Exported interface function. This function simply calls
1762 netlink_socket (). */
1763void
1764kernel_init ()
1765{
1766 unsigned long groups;
1767
paul7021c422003-07-15 12:52:22 +00001768 groups = RTMGRP_LINK | RTMGRP_IPV4_ROUTE | RTMGRP_IPV4_IFADDR;
paul718e3742002-12-13 20:15:29 +00001769#ifdef HAVE_IPV6
paul7021c422003-07-15 12:52:22 +00001770 groups |= RTMGRP_IPV6_ROUTE | RTMGRP_IPV6_IFADDR;
paul718e3742002-12-13 20:15:29 +00001771#endif /* HAVE_IPV6 */
1772 netlink_socket (&netlink, groups);
1773 netlink_socket (&netlink_cmd, 0);
1774
1775 /* Register kernel socket. */
1776 if (netlink.sock > 0)
paulb21b19c2003-06-15 01:28:29 +00001777 thread_add_read (zebrad.master, kernel_read, NULL, netlink.sock);
paul718e3742002-12-13 20:15:29 +00001778}