blob: 3db1271d01cfd71518d72bdc4d2ce3af84e3bdc7 [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/*
2 * Address linked list routine.
3 * Copyright (C) 1997, 98 Kunihiro Ishiguro
4 *
5 * This file is part of GNU Zebra.
6 *
7 * GNU Zebra is free software; you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License as published by the
9 * Free Software Foundation; either version 2, or (at your option) any
10 * later version.
11 *
12 * GNU Zebra is distributed in the hope that it will be useful, but
13 * WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with GNU Zebra; see the file COPYING. If not, write to the Free
19 * Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
20 * 02111-1307, USA.
21 */
22
23#include <zebra.h>
24
25#include "prefix.h"
26#include "linklist.h"
27#include "if.h"
28#include "table.h"
29#include "rib.h"
30#include "table.h"
31#include "log.h"
paul0752ef02005-11-03 12:35:21 +000032#include "memory.h"
paul718e3742002-12-13 20:15:29 +000033
34#include "zebra/zserv.h"
35#include "zebra/redistribute.h"
hassoeef1fe12004-10-03 18:46:08 +000036#include "zebra/interface.h"
paula1ac18c2005-06-28 17:17:12 +000037#include "zebra/connected.h"
Denis Ovsienko6ce80bd2007-11-12 14:55:01 +000038extern struct zebra_t zebrad;
paul718e3742002-12-13 20:15:29 +000039
paulca162182005-09-12 16:58:52 +000040/* withdraw a connected address */
41static void
42connected_withdraw (struct connected *ifc)
43{
44 if (! ifc)
45 return;
46
47 /* Update interface address information to protocol daemon. */
48 if (CHECK_FLAG (ifc->conf, ZEBRA_IFC_REAL))
49 {
50 zebra_interface_address_delete_update (ifc->ifp, ifc);
51
Christian Franke9db047f2013-01-24 14:04:44 +000052 if (ifc->address->family == AF_INET)
53 if_subnet_delete (ifc->ifp, ifc);
54
paulca162182005-09-12 16:58:52 +000055 if (ifc->address->family == AF_INET)
56 connected_down_ipv4 (ifc->ifp, ifc);
vincentaa2e32b2005-09-28 13:42:11 +000057#ifdef HAVE_IPV6
paulca162182005-09-12 16:58:52 +000058 else
59 connected_down_ipv6 (ifc->ifp, ifc);
vincentaa2e32b2005-09-28 13:42:11 +000060#endif
paulca162182005-09-12 16:58:52 +000061
62 UNSET_FLAG (ifc->conf, ZEBRA_IFC_REAL);
63 }
64
Andrew J. Schorr9c378512006-05-21 04:04:49 +000065 if (!CHECK_FLAG (ifc->conf, ZEBRA_IFC_CONFIGURED))
66 {
67 listnode_delete (ifc->ifp->connected, ifc);
68 connected_free (ifc);
69 }
paulca162182005-09-12 16:58:52 +000070}
71
72static void
73connected_announce (struct interface *ifp, struct connected *ifc)
74{
75 if (!ifc)
76 return;
77
78 listnode_add (ifp->connected, ifc);
79
80 /* Update interface address information to protocol daemon. */
81 if (! CHECK_FLAG (ifc->conf, ZEBRA_IFC_REAL))
82 {
83 if (ifc->address->family == AF_INET)
84 if_subnet_add (ifp, ifc);
85
86 SET_FLAG (ifc->conf, ZEBRA_IFC_REAL);
87
88 zebra_interface_address_add_update (ifp, ifc);
89
Stephen Hemmingerfd213252008-07-05 16:32:37 -070090 if (if_is_operative(ifp))
paulca162182005-09-12 16:58:52 +000091 {
92 if (ifc->address->family == AF_INET)
93 connected_up_ipv4 (ifp, ifc);
vincentaa2e32b2005-09-28 13:42:11 +000094#ifdef HAVE_IPV6
paulca162182005-09-12 16:58:52 +000095 else
96 connected_up_ipv6 (ifp, ifc);
vincentaa2e32b2005-09-28 13:42:11 +000097#endif
paulca162182005-09-12 16:58:52 +000098 }
99 }
100}
101
paul718e3742002-12-13 20:15:29 +0000102/* If same interface address is already exist... */
103struct connected *
paulca162182005-09-12 16:58:52 +0000104connected_check (struct interface *ifp, struct prefix *p)
paul718e3742002-12-13 20:15:29 +0000105{
106 struct connected *ifc;
hasso52dc7ee2004-09-23 19:18:23 +0000107 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000108
paul1eb8ef22005-04-07 07:30:20 +0000109 for (ALL_LIST_ELEMENTS_RO (ifp->connected, node, ifc))
110 if (prefix_same (ifc->address, p))
111 return ifc;
paul718e3742002-12-13 20:15:29 +0000112
paul718e3742002-12-13 20:15:29 +0000113 return NULL;
114}
115
Paul Jakma74ecdc92006-06-15 18:10:47 +0000116/* Check if two ifc's describe the same address */
117static int
118connected_same (struct connected *ifc1, struct connected *ifc2)
119{
120 if (ifc1->ifp != ifc2->ifp)
121 return 0;
122
123 if (ifc1->destination)
124 if (!ifc2->destination)
125 return 0;
126 if (ifc2->destination)
127 if (!ifc1->destination)
128 return 0;
129
130 if (ifc1->destination && ifc2->destination)
131 if (!prefix_same (ifc1->destination, ifc2->destination))
132 return 0;
133
134 if (ifc1->flags != ifc2->flags)
135 return 0;
136
137 return 1;
138}
139
140/* Handle implicit withdrawals of addresses, where a system ADDs an address
141 * to an interface which already has the same address configured.
142 *
143 * Returns the struct connected which must be announced to clients,
144 * or NULL if nothing to do.
145 */
146static struct connected *
147connected_implicit_withdraw (struct interface *ifp, struct connected *ifc)
148{
149 struct connected *current;
150
151 /* Check same connected route. */
152 if ((current = connected_check (ifp, (struct prefix *) ifc->address)))
153 {
154 if (CHECK_FLAG(current->conf, ZEBRA_IFC_CONFIGURED))
155 SET_FLAG(ifc->conf, ZEBRA_IFC_CONFIGURED);
156
157 /* Avoid spurious withdraws, this might be just the kernel 'reflecting'
158 * back an address we have already added.
159 */
David Young33b931e2007-04-16 05:54:02 +0000160 if (connected_same (current, ifc) && CHECK_FLAG(current->conf, ZEBRA_IFC_REAL))
Paul Jakma74ecdc92006-06-15 18:10:47 +0000161 {
162 /* nothing to do */
163 connected_free (ifc);
164 return NULL;
165 }
166
167 UNSET_FLAG(current->conf, ZEBRA_IFC_CONFIGURED);
168 connected_withdraw (current); /* implicit withdraw - freebsd does this */
169 }
170 return ifc;
171}
172
paul718e3742002-12-13 20:15:29 +0000173/* Called from if_up(). */
174void
175connected_up_ipv4 (struct interface *ifp, struct connected *ifc)
176{
177 struct prefix_ipv4 p;
paul718e3742002-12-13 20:15:29 +0000178
179 if (! CHECK_FLAG (ifc->conf, ZEBRA_IFC_REAL))
180 return;
181
Andrew J. Schorre4529632006-12-12 19:18:21 +0000182 PREFIX_COPY_IPV4(&p, CONNECTED_PREFIX(ifc));
paul718e3742002-12-13 20:15:29 +0000183
184 /* Apply mask to the network. */
185 apply_mask_ipv4 (&p);
186
187 /* In case of connected address is 0.0.0.0/0 we treat it tunnel
188 address. */
189 if (prefix_ipv4_any (&p))
190 return;
191
Paul Jakma7514fb72007-05-02 16:05:35 +0000192 rib_add_ipv4 (ZEBRA_ROUTE_CONNECT, 0, &p, NULL, NULL, ifp->ifindex,
G.Balajicddf3912011-11-26 21:59:32 +0400193 RT_TABLE_MAIN, ifp->metric, 0, SAFI_UNICAST);
paul718e3742002-12-13 20:15:29 +0000194
G.Balaji42cb6b62012-04-02 23:31:29 +0530195 rib_add_ipv4 (ZEBRA_ROUTE_CONNECT, 0, &p, NULL, NULL, ifp->ifindex,
196 RT_TABLE_MAIN, ifp->metric, 0, SAFI_MULTICAST);
197
paul718e3742002-12-13 20:15:29 +0000198 rib_update ();
199}
200
201/* Add connected IPv4 route to the interface. */
202void
203connected_add_ipv4 (struct interface *ifp, int flags, struct in_addr *addr,
pauld06b2a62005-10-11 03:53:54 +0000204 u_char prefixlen, struct in_addr *broad,
205 const char *label)
paul718e3742002-12-13 20:15:29 +0000206{
207 struct prefix_ipv4 *p;
208 struct connected *ifc;
paul718e3742002-12-13 20:15:29 +0000209
210 /* Make connected structure. */
211 ifc = connected_new ();
212 ifc->ifp = ifp;
213 ifc->flags = flags;
214
215 /* Allocate new connected address. */
216 p = prefix_ipv4_new ();
217 p->family = AF_INET;
218 p->prefix = *addr;
219 p->prefixlen = prefixlen;
220 ifc->address = (struct prefix *) p;
paulca162182005-09-12 16:58:52 +0000221
Andrew J. Schorre4529632006-12-12 19:18:21 +0000222 /* If there is broadcast or peer address. */
paul718e3742002-12-13 20:15:29 +0000223 if (broad)
224 {
225 p = prefix_ipv4_new ();
226 p->family = AF_INET;
227 p->prefix = *broad;
Andrew J. Schorre4529632006-12-12 19:18:21 +0000228 p->prefixlen = prefixlen;
paul718e3742002-12-13 20:15:29 +0000229 ifc->destination = (struct prefix *) p;
hasso3fb9cd62004-10-19 19:44:43 +0000230
231 /* validate the destination address */
Andrew J. Schorre4529632006-12-12 19:18:21 +0000232 if (CONNECTED_PEER(ifc))
hasso3fb9cd62004-10-19 19:44:43 +0000233 {
234 if (IPV4_ADDR_SAME(addr,broad))
Andrew J. Schorre4529632006-12-12 19:18:21 +0000235 zlog_warn("warning: interface %s has same local and peer "
hasso3fb9cd62004-10-19 19:44:43 +0000236 "address %s, routing protocols may malfunction",
237 ifp->name,inet_ntoa(*addr));
hasso3fb9cd62004-10-19 19:44:43 +0000238 }
239 else
240 {
241 if (broad->s_addr != ipv4_broadcast_addr(addr->s_addr,prefixlen))
242 {
243 char buf[2][INET_ADDRSTRLEN];
244 struct in_addr bcalc;
245 bcalc.s_addr = ipv4_broadcast_addr(addr->s_addr,prefixlen);
246 zlog_warn("warning: interface %s broadcast addr %s/%d != "
247 "calculated %s, routing protocols may malfunction",
248 ifp->name,
249 inet_ntop (AF_INET, broad, buf[0], sizeof(buf[0])),
250 prefixlen,
251 inet_ntop (AF_INET, &bcalc, buf[1], sizeof(buf[1])));
252 }
253 }
254
paul718e3742002-12-13 20:15:29 +0000255 }
hasso3fb9cd62004-10-19 19:44:43 +0000256 else
Andrew J. Schorre4529632006-12-12 19:18:21 +0000257 {
258 if (CHECK_FLAG(ifc->flags, ZEBRA_IFA_PEER))
259 {
260 zlog_warn("warning: %s called for interface %s "
261 "with peer flag set, but no peer address supplied",
262 __func__, ifp->name);
263 UNSET_FLAG(ifc->flags, ZEBRA_IFA_PEER);
264 }
265
266 /* no broadcast or destination address was supplied */
267 if ((prefixlen == IPV4_MAX_PREFIXLEN) && if_is_pointopoint(ifp))
268 zlog_warn("warning: PtP interface %s with addr %s/%d needs a "
269 "peer address",ifp->name,inet_ntoa(*addr),prefixlen);
270 }
paul718e3742002-12-13 20:15:29 +0000271
272 /* Label of this address. */
273 if (label)
paul0752ef02005-11-03 12:35:21 +0000274 ifc->label = XSTRDUP (MTYPE_CONNECTED_LABEL, label);
paul718e3742002-12-13 20:15:29 +0000275
Paul Jakma74ecdc92006-06-15 18:10:47 +0000276 /* nothing to do? */
277 if ((ifc = connected_implicit_withdraw (ifp, ifc)) == NULL)
278 return;
paulca162182005-09-12 16:58:52 +0000279
280 connected_announce (ifp, ifc);
paul718e3742002-12-13 20:15:29 +0000281}
282
283void
284connected_down_ipv4 (struct interface *ifp, struct connected *ifc)
285{
286 struct prefix_ipv4 p;
paul718e3742002-12-13 20:15:29 +0000287
288 if (! CHECK_FLAG (ifc->conf, ZEBRA_IFC_REAL))
289 return;
290
Andrew J. Schorre4529632006-12-12 19:18:21 +0000291 PREFIX_COPY_IPV4(&p, CONNECTED_PREFIX(ifc));
paul718e3742002-12-13 20:15:29 +0000292
293 /* Apply mask to the network. */
294 apply_mask_ipv4 (&p);
295
296 /* In case of connected address is 0.0.0.0/0 we treat it tunnel
297 address. */
298 if (prefix_ipv4_any (&p))
299 return;
300
Denis Ovsienko6ce80bd2007-11-12 14:55:01 +0000301 /* Same logic as for connected_up_ipv4(): push the changes into the head. */
G.Balajicddf3912011-11-26 21:59:32 +0400302 rib_delete_ipv4 (ZEBRA_ROUTE_CONNECT, 0, &p, NULL, ifp->ifindex, 0, SAFI_UNICAST);
paul718e3742002-12-13 20:15:29 +0000303
G.Balaji42cb6b62012-04-02 23:31:29 +0530304 rib_delete_ipv4 (ZEBRA_ROUTE_CONNECT, 0, &p, NULL, ifp->ifindex, 0, SAFI_MULTICAST);
305
paul718e3742002-12-13 20:15:29 +0000306 rib_update ();
307}
308
309/* Delete connected IPv4 route to the interface. */
310void
311connected_delete_ipv4 (struct interface *ifp, int flags, struct in_addr *addr,
paul0752ef02005-11-03 12:35:21 +0000312 u_char prefixlen, struct in_addr *broad)
paul718e3742002-12-13 20:15:29 +0000313{
314 struct prefix_ipv4 p;
315 struct connected *ifc;
316
317 memset (&p, 0, sizeof (struct prefix_ipv4));
318 p.family = AF_INET;
319 p.prefix = *addr;
320 p.prefixlen = prefixlen;
321
paulca162182005-09-12 16:58:52 +0000322 ifc = connected_check (ifp, (struct prefix *) &p);
paul718e3742002-12-13 20:15:29 +0000323 if (! ifc)
324 return;
paulca162182005-09-12 16:58:52 +0000325
326 connected_withdraw (ifc);
Stephen Hemminger90d2ab02009-04-29 21:54:59 -0700327
328 rib_update();
paul718e3742002-12-13 20:15:29 +0000329}
330
331#ifdef HAVE_IPV6
paul718e3742002-12-13 20:15:29 +0000332void
333connected_up_ipv6 (struct interface *ifp, struct connected *ifc)
334{
335 struct prefix_ipv6 p;
paul718e3742002-12-13 20:15:29 +0000336
337 if (! CHECK_FLAG (ifc->conf, ZEBRA_IFC_REAL))
338 return;
339
Andrew J. Schorre4529632006-12-12 19:18:21 +0000340 PREFIX_COPY_IPV6(&p, CONNECTED_PREFIX(ifc));
paul718e3742002-12-13 20:15:29 +0000341
342 /* Apply mask to the network. */
343 apply_mask_ipv6 (&p);
344
hasso726f9b22003-05-25 21:04:54 +0000345#if ! defined (MUSICA) && ! defined (LINUX)
346 /* XXX: It is already done by rib_bogus_ipv6 within rib_add_ipv6 */
paul718e3742002-12-13 20:15:29 +0000347 if (IN6_IS_ADDR_UNSPECIFIED (&p.prefix))
348 return;
hasso726f9b22003-05-25 21:04:54 +0000349#endif
paul718e3742002-12-13 20:15:29 +0000350
Mathieu Goessensd13c3b42009-06-23 15:59:45 +0100351 rib_add_ipv6 (ZEBRA_ROUTE_CONNECT, 0, &p, NULL, ifp->ifindex, RT_TABLE_MAIN,
G.Balajif768f362011-11-26 22:10:39 +0400352 ifp->metric, 0, SAFI_UNICAST);
paul718e3742002-12-13 20:15:29 +0000353
354 rib_update ();
355}
356
357/* Add connected IPv6 route to the interface. */
358void
Andrew J. Schorre4529632006-12-12 19:18:21 +0000359connected_add_ipv6 (struct interface *ifp, int flags, struct in6_addr *addr,
paul89368d92005-11-26 09:14:07 +0000360 u_char prefixlen, struct in6_addr *broad,
361 const char *label)
paul718e3742002-12-13 20:15:29 +0000362{
363 struct prefix_ipv6 *p;
364 struct connected *ifc;
paul718e3742002-12-13 20:15:29 +0000365
366 /* Make connected structure. */
367 ifc = connected_new ();
368 ifc->ifp = ifp;
Andrew J. Schorre4529632006-12-12 19:18:21 +0000369 ifc->flags = flags;
paul718e3742002-12-13 20:15:29 +0000370
371 /* Allocate new connected address. */
372 p = prefix_ipv6_new ();
373 p->family = AF_INET6;
374 IPV6_ADDR_COPY (&p->prefix, addr);
375 p->prefixlen = prefixlen;
376 ifc->address = (struct prefix *) p;
377
Andrew J. Schorre4529632006-12-12 19:18:21 +0000378 /* If there is broadcast or peer address. */
paul718e3742002-12-13 20:15:29 +0000379 if (broad)
380 {
Andrew J. Schorre4529632006-12-12 19:18:21 +0000381 if (IN6_IS_ADDR_UNSPECIFIED(broad))
382 zlog_warn("warning: %s called for interface %s with unspecified "
383 "destination address; ignoring!", __func__, ifp->name);
384 else
385 {
386 p = prefix_ipv6_new ();
387 p->family = AF_INET6;
388 IPV6_ADDR_COPY (&p->prefix, broad);
389 p->prefixlen = prefixlen;
390 ifc->destination = (struct prefix *) p;
391 }
392 }
393 if (CHECK_FLAG(ifc->flags, ZEBRA_IFA_PEER) && !ifc->destination)
394 {
395 zlog_warn("warning: %s called for interface %s "
396 "with peer flag set, but no peer address supplied",
397 __func__, ifp->name);
398 UNSET_FLAG(ifc->flags, ZEBRA_IFA_PEER);
paul718e3742002-12-13 20:15:29 +0000399 }
400
paul0752ef02005-11-03 12:35:21 +0000401 /* Label of this address. */
402 if (label)
403 ifc->label = XSTRDUP (MTYPE_CONNECTED_LABEL, label);
404
Paul Jakma74ecdc92006-06-15 18:10:47 +0000405 if ((ifc = connected_implicit_withdraw (ifp, ifc)) == NULL)
406 return;
paulca162182005-09-12 16:58:52 +0000407
408 connected_announce (ifp, ifc);
paul718e3742002-12-13 20:15:29 +0000409}
410
411void
412connected_down_ipv6 (struct interface *ifp, struct connected *ifc)
413{
414 struct prefix_ipv6 p;
paul718e3742002-12-13 20:15:29 +0000415
416 if (! CHECK_FLAG (ifc->conf, ZEBRA_IFC_REAL))
417 return;
418
Andrew J. Schorre4529632006-12-12 19:18:21 +0000419 PREFIX_COPY_IPV6(&p, CONNECTED_PREFIX(ifc));
paul718e3742002-12-13 20:15:29 +0000420
421 apply_mask_ipv6 (&p);
422
423 if (IN6_IS_ADDR_UNSPECIFIED (&p.prefix))
424 return;
425
G.Balajif768f362011-11-26 22:10:39 +0400426 rib_delete_ipv6 (ZEBRA_ROUTE_CONNECT, 0, &p, NULL, ifp->ifindex, 0, SAFI_UNICAST);
paul718e3742002-12-13 20:15:29 +0000427
428 rib_update ();
429}
430
431void
432connected_delete_ipv6 (struct interface *ifp, struct in6_addr *address,
hassofce954f2004-10-07 20:29:24 +0000433 u_char prefixlen, struct in6_addr *broad)
paul718e3742002-12-13 20:15:29 +0000434{
435 struct prefix_ipv6 p;
436 struct connected *ifc;
437
438 memset (&p, 0, sizeof (struct prefix_ipv6));
439 p.family = AF_INET6;
440 memcpy (&p.prefix, address, sizeof (struct in6_addr));
441 p.prefixlen = prefixlen;
442
paulca162182005-09-12 16:58:52 +0000443 ifc = connected_check (ifp, (struct prefix *) &p);
paul718e3742002-12-13 20:15:29 +0000444 if (! ifc)
445 return;
446
paulca162182005-09-12 16:58:52 +0000447 connected_withdraw (ifc);
Stephen Hemminger90d2ab02009-04-29 21:54:59 -0700448
449 rib_update();
paul718e3742002-12-13 20:15:29 +0000450}
451#endif /* HAVE_IPV6 */