blob: f5f49f644eafff7bfbb6f143a85a1c7710d9c19c [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/*
2 * Zebra connect library for OSPFd
3 * Copyright (C) 1997, 98, 99, 2000 Kunihiro Ishiguro, Toshiaki Takada
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
19 * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
20 * Boston, MA 02111-1307, USA.
21 */
22
23#include <zebra.h>
24
25#include "thread.h"
26#include "command.h"
27#include "network.h"
28#include "prefix.h"
29#include "routemap.h"
30#include "table.h"
31#include "stream.h"
32#include "memory.h"
33#include "zclient.h"
34#include "filter.h"
hassodd669bb2004-05-10 07:43:59 +000035#include "plist.h"
paul718e3742002-12-13 20:15:29 +000036#include "log.h"
37
38#include "ospfd/ospfd.h"
39#include "ospfd/ospf_interface.h"
40#include "ospfd/ospf_ism.h"
41#include "ospfd/ospf_asbr.h"
42#include "ospfd/ospf_asbr.h"
43#include "ospfd/ospf_abr.h"
44#include "ospfd/ospf_lsa.h"
45#include "ospfd/ospf_dump.h"
46#include "ospfd/ospf_route.h"
47#include "ospfd/ospf_zebra.h"
48#ifdef HAVE_SNMP
49#include "ospfd/ospf_snmp.h"
50#endif /* HAVE_SNMP */
51
52/* Zebra structure to hold current status. */
53struct zclient *zclient = NULL;
54
55/* For registering threads. */
56extern struct thread_master *master;
hasso18a6dce2004-10-03 18:18:34 +000057struct in_addr router_id_zebra;
58
59/* Router-id update message from zebra. */
paul4dadc292005-05-06 21:37:42 +000060static int
hasso18a6dce2004-10-03 18:18:34 +000061ospf_router_id_update_zebra (int command, struct zclient *zclient,
62 zebra_size_t length)
63{
64 struct ospf *ospf;
65 struct prefix router_id;
66 zebra_router_id_update_read(zclient->ibuf,&router_id);
67
Andrew J. Schorr7f643eb2006-11-30 16:17:02 +000068 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
69 {
70 char buf[128];
71 prefix2str(&router_id, buf, sizeof(buf));
72 zlog_debug("Zebra rcvd: router id update %s", buf);
73 }
74
hasso18a6dce2004-10-03 18:18:34 +000075 router_id_zebra = router_id.u.prefix4;
76
77 ospf = ospf_lookup ();
paulb29800a2005-11-20 14:50:45 +000078
hasso18a6dce2004-10-03 18:18:34 +000079 if (ospf != NULL)
paulb29800a2005-11-20 14:50:45 +000080 ospf_router_id_update (ospf);
81
hasso18a6dce2004-10-03 18:18:34 +000082 return 0;
83}
paul718e3742002-12-13 20:15:29 +000084
85/* Inteface addition message from zebra. */
paul4dadc292005-05-06 21:37:42 +000086static int
paul718e3742002-12-13 20:15:29 +000087ospf_interface_add (int command, struct zclient *zclient, zebra_size_t length)
88{
89 struct interface *ifp;
90
91 ifp = zebra_interface_add_read (zclient->ibuf);
92
93 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
Stephen Hemminger30d20592009-07-28 11:58:51 +010094 zlog_debug ("Zebra: interface add %s index %d flags %llx metric %d mtu %d",
95 ifp->name, ifp->ifindex, (unsigned long long)ifp->flags,
96 ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +000097
paulcf795c52003-06-19 02:13:25 +000098 assert (ifp->info);
paulf2c80652002-12-13 21:44:27 +000099
paul718e3742002-12-13 20:15:29 +0000100 if (!OSPF_IF_PARAM_CONFIGURED (IF_DEF_PARAMS (ifp), type))
101 {
102 SET_IF_PARAM (IF_DEF_PARAMS (ifp), type);
ajsbc18d612004-12-15 15:07:19 +0000103 IF_DEF_PARAMS (ifp)->type = ospf_default_iftype(ifp);
paul718e3742002-12-13 20:15:29 +0000104 }
105
Joakim Tjernlunda49eb302008-09-02 19:06:31 +0100106 ospf_if_update (NULL, ifp);
paul718e3742002-12-13 20:15:29 +0000107
108#ifdef HAVE_SNMP
109 ospf_snmp_if_update (ifp);
110#endif /* HAVE_SNMP */
111
112 return 0;
113}
114
paul4dadc292005-05-06 21:37:42 +0000115static int
paul718e3742002-12-13 20:15:29 +0000116ospf_interface_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000117 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000118{
119 struct interface *ifp;
120 struct stream *s;
121 struct route_node *rn;
122
paulcf795c52003-06-19 02:13:25 +0000123 s = zclient->ibuf;
paul718e3742002-12-13 20:15:29 +0000124 /* zebra_interface_state_read() updates interface structure in iflist */
125 ifp = zebra_interface_state_read (s);
126
127 if (ifp == NULL)
128 return 0;
129
130 if (if_is_up (ifp))
131 zlog_warn ("Zebra: got delete of %s, but interface is still up",
paulcf795c52003-06-19 02:13:25 +0000132 ifp->name);
133
paul718e3742002-12-13 20:15:29 +0000134 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000135 zlog_debug
Stephen Hemminger30d20592009-07-28 11:58:51 +0100136 ("Zebra: interface delete %s index %d flags %lld metric %d mtu %d",
paulcf795c52003-06-19 02:13:25 +0000137 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +0000138
139#ifdef HAVE_SNMP
140 ospf_snmp_if_delete (ifp);
141#endif /* HAVE_SNMP */
142
143 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
144 if (rn->info)
145 ospf_if_free ((struct ospf_interface *) rn->info);
146
ajsd2fc8892005-04-02 18:38:43 +0000147 ifp->ifindex = IFINDEX_INTERNAL;
paul718e3742002-12-13 20:15:29 +0000148 return 0;
149}
150
ajsd2fc8892005-04-02 18:38:43 +0000151static struct interface *
paul718e3742002-12-13 20:15:29 +0000152zebra_interface_if_lookup (struct stream *s)
153{
ajs21fefa92005-04-02 23:16:41 +0000154 char ifname_tmp[INTERFACE_NAMSIZ];
paul718e3742002-12-13 20:15:29 +0000155
156 /* Read interface name. */
157 stream_get (ifname_tmp, s, INTERFACE_NAMSIZ);
158
ajsd2fc8892005-04-02 18:38:43 +0000159 /* And look it up. */
ajs21fefa92005-04-02 23:16:41 +0000160 return if_lookup_by_name_len(ifname_tmp,
161 strnlen(ifname_tmp, INTERFACE_NAMSIZ));
paul718e3742002-12-13 20:15:29 +0000162}
163
paul4dadc292005-05-06 21:37:42 +0000164static int
paul718e3742002-12-13 20:15:29 +0000165ospf_interface_state_up (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000166 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000167{
168 struct interface *ifp;
paul718e3742002-12-13 20:15:29 +0000169 struct ospf_interface *oi;
170 struct route_node *rn;
paulcf795c52003-06-19 02:13:25 +0000171
paul718e3742002-12-13 20:15:29 +0000172 ifp = zebra_interface_if_lookup (zclient->ibuf);
173
174 if (ifp == NULL)
175 return 0;
176
177 /* Interface is already up. */
paul2e3b2e42002-12-13 21:03:13 +0000178 if (if_is_operative (ifp))
paul718e3742002-12-13 20:15:29 +0000179 {
180 /* Temporarily keep ifp values. */
ajsa608bbf2005-03-29 17:03:49 +0000181 struct interface if_tmp;
paul718e3742002-12-13 20:15:29 +0000182 memcpy (&if_tmp, ifp, sizeof (struct interface));
183
184 zebra_interface_if_set_value (zclient->ibuf, ifp);
185
186 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000187 zlog_debug ("Zebra: Interface[%s] state update.", ifp->name);
paul718e3742002-12-13 20:15:29 +0000188
189 if (if_tmp.bandwidth != ifp->bandwidth)
paulcf795c52003-06-19 02:13:25 +0000190 {
191 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000192 zlog_debug ("Zebra: Interface[%s] bandwidth change %d -> %d.",
paulcf795c52003-06-19 02:13:25 +0000193 ifp->name, if_tmp.bandwidth, ifp->bandwidth);
paul718e3742002-12-13 20:15:29 +0000194
paulcf795c52003-06-19 02:13:25 +0000195 ospf_if_recalculate_output_cost (ifp);
196 }
ajsa608bbf2005-03-29 17:03:49 +0000197
198 if (if_tmp.mtu != ifp->mtu)
199 {
200 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
201 zlog_debug ("Zebra: Interface[%s] MTU change %u -> %u.",
202 ifp->name, if_tmp.mtu, ifp->mtu);
203
204 /* Must reset the interface (simulate down/up) when MTU changes. */
205 ospf_if_reset(ifp);
206 }
paul718e3742002-12-13 20:15:29 +0000207 return 0;
208 }
paulcf795c52003-06-19 02:13:25 +0000209
paul718e3742002-12-13 20:15:29 +0000210 zebra_interface_if_set_value (zclient->ibuf, ifp);
paulcf795c52003-06-19 02:13:25 +0000211
paul718e3742002-12-13 20:15:29 +0000212 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000213 zlog_debug ("Zebra: Interface[%s] state change to up.", ifp->name);
paulcf795c52003-06-19 02:13:25 +0000214
215 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +0000216 {
paulcf795c52003-06-19 02:13:25 +0000217 if ((oi = rn->info) == NULL)
218 continue;
219
paul718e3742002-12-13 20:15:29 +0000220 ospf_if_up (oi);
221 }
paulcf795c52003-06-19 02:13:25 +0000222
paul718e3742002-12-13 20:15:29 +0000223 return 0;
224}
225
paul4dadc292005-05-06 21:37:42 +0000226static int
paul718e3742002-12-13 20:15:29 +0000227ospf_interface_state_down (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000228 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000229{
230 struct interface *ifp;
231 struct ospf_interface *oi;
232 struct route_node *node;
233
234 ifp = zebra_interface_state_read (zclient->ibuf);
235
236 if (ifp == NULL)
237 return 0;
238
239 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000240 zlog_debug ("Zebra: Interface[%s] state change to down.", ifp->name);
paul718e3742002-12-13 20:15:29 +0000241
paulcf795c52003-06-19 02:13:25 +0000242 for (node = route_top (IF_OIFS (ifp)); node; node = route_next (node))
paul718e3742002-12-13 20:15:29 +0000243 {
paulcf795c52003-06-19 02:13:25 +0000244 if ((oi = node->info) == NULL)
245 continue;
paul718e3742002-12-13 20:15:29 +0000246 ospf_if_down (oi);
247 }
248
249 return 0;
250}
251
paul4dadc292005-05-06 21:37:42 +0000252static int
paul718e3742002-12-13 20:15:29 +0000253ospf_interface_address_add (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000254 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000255{
256 struct connected *c;
257
paul0a589352004-05-08 11:48:26 +0000258 c = zebra_interface_address_read (command, zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000259
260 if (c == NULL)
261 return 0;
262
Andrew J. Schorr7f643eb2006-11-30 16:17:02 +0000263 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
264 {
265 char buf[128];
266 prefix2str(c->address, buf, sizeof(buf));
267 zlog_debug("Zebra: interface %s address add %s", c->ifp->name, buf);
268 }
269
Joakim Tjernlunda49eb302008-09-02 19:06:31 +0100270 ospf_if_update (NULL, c->ifp);
paul718e3742002-12-13 20:15:29 +0000271
272#ifdef HAVE_SNMP
273 ospf_snmp_if_update (c->ifp);
274#endif /* HAVE_SNMP */
275
276 return 0;
277}
278
paul4dadc292005-05-06 21:37:42 +0000279static int
paul718e3742002-12-13 20:15:29 +0000280ospf_interface_address_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000281 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000282{
283 struct connected *c;
284 struct interface *ifp;
285 struct ospf_interface *oi;
286 struct route_node *rn;
287 struct prefix p;
288
paul0a589352004-05-08 11:48:26 +0000289 c = zebra_interface_address_read (command, zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000290
291 if (c == NULL)
292 return 0;
293
Andrew J. Schorr7f643eb2006-11-30 16:17:02 +0000294 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
295 {
296 char buf[128];
297 prefix2str(c->address, buf, sizeof(buf));
298 zlog_debug("Zebra: interface %s address delete %s", c->ifp->name, buf);
299 }
300
paul718e3742002-12-13 20:15:29 +0000301 ifp = c->ifp;
302 p = *c->address;
303 p.prefixlen = IPV4_MAX_PREFIXLEN;
304
305 rn = route_node_lookup (IF_OIFS (ifp), &p);
paulcf795c52003-06-19 02:13:25 +0000306 if (!rn)
paul98429f62006-01-10 22:11:54 +0000307 {
308 connected_free (c);
309 return 0;
310 }
paul718e3742002-12-13 20:15:29 +0000311
312 assert (rn->info);
313 oi = rn->info;
paulcf795c52003-06-19 02:13:25 +0000314
paul718e3742002-12-13 20:15:29 +0000315 /* Call interface hook functions to clean up */
316 ospf_if_free (oi);
paulcf795c52003-06-19 02:13:25 +0000317
paul718e3742002-12-13 20:15:29 +0000318#ifdef HAVE_SNMP
319 ospf_snmp_if_update (c->ifp);
320#endif /* HAVE_SNMP */
321
322 connected_free (c);
323
paul718e3742002-12-13 20:15:29 +0000324 return 0;
325}
paul72357f22003-06-19 02:11:23 +0000326
paul718e3742002-12-13 20:15:29 +0000327void
328ospf_zebra_add (struct prefix_ipv4 *p, struct ospf_route *or)
329{
330 u_char message;
331 u_char distance;
332 u_char flags;
333 int psize;
334 struct stream *s;
335 struct ospf_path *path;
hasso52dc7ee2004-09-23 19:18:23 +0000336 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000337
338 if (zclient->redist[ZEBRA_ROUTE_OSPF])
339 {
340 message = 0;
341 flags = 0;
342
343 /* OSPF pass nexthop and metric */
344 SET_FLAG (message, ZAPI_MESSAGE_NEXTHOP);
345 SET_FLAG (message, ZAPI_MESSAGE_METRIC);
346
347 /* Distance value. */
348 distance = ospf_distance_apply (p, or);
349 if (distance)
paul72357f22003-06-19 02:11:23 +0000350 SET_FLAG (message, ZAPI_MESSAGE_DISTANCE);
paul718e3742002-12-13 20:15:29 +0000351
352 /* Make packet. */
353 s = zclient->obuf;
354 stream_reset (s);
355
paul718e3742002-12-13 20:15:29 +0000356 /* Put command, type, flags, message. */
paulc6371712006-01-17 17:49:53 +0000357 zclient_create_header (s, ZEBRA_IPV4_ROUTE_ADD);
paul718e3742002-12-13 20:15:29 +0000358 stream_putc (s, ZEBRA_ROUTE_OSPF);
359 stream_putc (s, flags);
360 stream_putc (s, message);
Denis Ovsienkob4e45f62011-12-05 16:35:14 +0400361 stream_putw (s, SAFI_UNICAST);
paulcf795c52003-06-19 02:13:25 +0000362
paul718e3742002-12-13 20:15:29 +0000363 /* Put prefix information. */
364 psize = PSIZE (p->prefixlen);
365 stream_putc (s, p->prefixlen);
paulcf795c52003-06-19 02:13:25 +0000366 stream_write (s, (u_char *) & p->prefix, psize);
paul718e3742002-12-13 20:15:29 +0000367
368 /* Nexthop count. */
paul96735ee2003-08-10 02:51:22 +0000369 stream_putc (s, or->paths->count);
paul718e3742002-12-13 20:15:29 +0000370
371 /* Nexthop, ifindex, distance and metric information. */
paul1eb8ef22005-04-07 07:30:20 +0000372 for (ALL_LIST_ELEMENTS_RO (or->paths, node, path))
paul72357f22003-06-19 02:11:23 +0000373 {
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200374 if (path->nexthop.s_addr != INADDR_ANY &&
375 path->ifindex != 0)
376 {
377 stream_putc (s, ZEBRA_NEXTHOP_IPV4_IFINDEX);
378 stream_put_in_addr (s, &path->nexthop);
379 stream_putl (s, path->ifindex);
380 }
381 else if (path->nexthop.s_addr != INADDR_ANY)
paulcf795c52003-06-19 02:13:25 +0000382 {
383 stream_putc (s, ZEBRA_NEXTHOP_IPV4);
384 stream_put_in_addr (s, &path->nexthop);
385 }
386 else
387 {
388 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
Joakim Tjernlunda8ba8472009-07-27 12:42:34 +0200389 if (path->ifindex)
390 stream_putl (s, path->ifindex);
paulcf795c52003-06-19 02:13:25 +0000391 else
392 stream_putl (s, 0);
393 }
paul72357f22003-06-19 02:11:23 +0000394
395 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
396 {
Andrew J. Schorr56b3ea02007-03-14 20:21:43 +0000397 char buf[2][INET_ADDRSTRLEN];
398 zlog_debug("Zebra: Route add %s/%d nexthop %s",
399 inet_ntop(AF_INET, &p->prefix,
400 buf[0], sizeof(buf[0])),
401 p->prefixlen,
402 inet_ntop(AF_INET, &path->nexthop,
403 buf[1], sizeof(buf[1])));
paulcf795c52003-06-19 02:13:25 +0000404 }
405 }
paul718e3742002-12-13 20:15:29 +0000406
407 if (CHECK_FLAG (message, ZAPI_MESSAGE_DISTANCE))
paulcf795c52003-06-19 02:13:25 +0000408 stream_putc (s, distance);
paul718e3742002-12-13 20:15:29 +0000409 if (CHECK_FLAG (message, ZAPI_MESSAGE_METRIC))
paulcf795c52003-06-19 02:13:25 +0000410 {
411 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL)
412 stream_putl (s, or->cost + or->u.ext.type2_cost);
413 else if (or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
414 stream_putl (s, or->u.ext.type2_cost);
415 else
416 stream_putl (s, or->cost);
417 }
paul718e3742002-12-13 20:15:29 +0000418
419 stream_putw_at (s, 0, stream_get_endp (s));
420
ajs634f9ea2005-04-11 15:51:40 +0000421 zclient_send_message(zclient);
paul718e3742002-12-13 20:15:29 +0000422 }
423}
424
425void
426ospf_zebra_delete (struct prefix_ipv4 *p, struct ospf_route *or)
427{
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200428 u_char message;
429 u_char distance;
430 u_char flags;
431 int psize;
432 struct stream *s;
paul72357f22003-06-19 02:11:23 +0000433 struct ospf_path *path;
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200434 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000435
436 if (zclient->redist[ZEBRA_ROUTE_OSPF])
437 {
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200438 message = 0;
439 flags = 0;
440 /* Distance value. */
441 distance = ospf_distance_apply (p, or);
442 /* Make packet. */
443 s = zclient->obuf;
444 stream_reset (s);
paul718e3742002-12-13 20:15:29 +0000445
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200446 /* Put command, type, flags, message. */
447 zclient_create_header (s, ZEBRA_IPV4_ROUTE_DELETE);
448 stream_putc (s, ZEBRA_ROUTE_OSPF);
449 stream_putc (s, flags);
450 stream_putc (s, message);
451 stream_putw (s, SAFI_UNICAST);
paul72357f22003-06-19 02:11:23 +0000452
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200453 /* Put prefix information. */
454 psize = PSIZE (p->prefixlen);
455 stream_putc (s, p->prefixlen);
456 stream_write (s, (u_char *) & p->prefix, psize);
paul72357f22003-06-19 02:11:23 +0000457
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200458 /* Nexthop count. */
459 stream_putc (s, or->paths->count);
460
461 /* Nexthop, ifindex, distance and metric information. */
462 for (ALL_LIST_ELEMENTS_RO (or->paths, node, path))
463 {
464 if (path->nexthop.s_addr != INADDR_ANY &&
465 path->ifindex != 0)
466 {
467 stream_putc (s, ZEBRA_NEXTHOP_IPV4_IFINDEX);
468 stream_put_in_addr (s, &path->nexthop);
469 stream_putl (s, path->ifindex);
470 }
471 else if (path->nexthop.s_addr != INADDR_ANY)
472 {
473 stream_putc (s, ZEBRA_NEXTHOP_IPV4);
474 stream_put_in_addr (s, &path->nexthop);
475 }
476 else
477 {
478 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
479 stream_putl (s, path->ifindex);
480 }
481
482 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
483 {
Andrew J. Schorr56b3ea02007-03-14 20:21:43 +0000484 char buf[2][INET_ADDRSTRLEN];
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200485 zlog_debug("Zebra: Route add %s/%d nexthop %s",
486 inet_ntop(AF_INET, &p->prefix,
487 buf[0], sizeof(buf[0])),
Andrew J. Schorr56b3ea02007-03-14 20:21:43 +0000488 p->prefixlen,
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200489 inet_ntop(AF_INET, &path->nexthop,
Andrew J. Schorr56b3ea02007-03-14 20:21:43 +0000490 buf[1], sizeof(buf[1])));
Joakim Tjernlundba281d32012-07-07 17:06:13 +0200491 }
492 }
493
494 if (CHECK_FLAG (message, ZAPI_MESSAGE_DISTANCE))
495 stream_putc (s, distance);
496 if (CHECK_FLAG (message, ZAPI_MESSAGE_METRIC))
497 {
498 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL)
499 stream_putl (s, or->cost + or->u.ext.type2_cost);
500 else if (or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
501 stream_putl (s, or->u.ext.type2_cost);
502 else
503 stream_putl (s, or->cost);
504 }
505
506 stream_putw_at (s, 0, stream_get_endp (s));
507
508 zclient_send_message(zclient);
paul718e3742002-12-13 20:15:29 +0000509 }
510}
511
512void
513ospf_zebra_add_discard (struct prefix_ipv4 *p)
514{
515 struct zapi_ipv4 api;
516
517 if (zclient->redist[ZEBRA_ROUTE_OSPF])
518 {
519 api.type = ZEBRA_ROUTE_OSPF;
520 api.flags = ZEBRA_FLAG_BLACKHOLE;
521 api.message = 0;
Denis Ovsienkob4e45f62011-12-05 16:35:14 +0400522 api.safi = SAFI_UNICAST;
paul718e3742002-12-13 20:15:29 +0000523 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
524 api.nexthop_num = 0;
525 api.ifindex_num = 0;
526
paul0a589352004-05-08 11:48:26 +0000527 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_ADD, zclient, p, &api);
Andrew J. Schorr7f643eb2006-11-30 16:17:02 +0000528
529 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
530 zlog_debug ("Zebra: Route add discard %s/%d",
531 inet_ntoa (p->prefix), p->prefixlen);
paul718e3742002-12-13 20:15:29 +0000532 }
533}
534
535void
536ospf_zebra_delete_discard (struct prefix_ipv4 *p)
537{
538 struct zapi_ipv4 api;
539
540 if (zclient->redist[ZEBRA_ROUTE_OSPF])
541 {
542 api.type = ZEBRA_ROUTE_OSPF;
543 api.flags = ZEBRA_FLAG_BLACKHOLE;
544 api.message = 0;
Denis Ovsienkob4e45f62011-12-05 16:35:14 +0400545 api.safi = SAFI_UNICAST;
paul718e3742002-12-13 20:15:29 +0000546 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
547 api.nexthop_num = 0;
548 api.ifindex_num = 0;
549
paul0a589352004-05-08 11:48:26 +0000550 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
paul72357f22003-06-19 02:11:23 +0000551
552 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000553 zlog_debug ("Zebra: Route delete discard %s/%d",
paulcf795c52003-06-19 02:13:25 +0000554 inet_ntoa (p->prefix), p->prefixlen);
paul72357f22003-06-19 02:11:23 +0000555
paul718e3742002-12-13 20:15:29 +0000556 }
557}
558
559int
560ospf_is_type_redistributed (int type)
561{
562 return (DEFAULT_ROUTE_TYPE (type)) ?
563 zclient->default_information : zclient->redist[type];
564}
565
566int
paul020709f2003-04-04 02:44:16 +0000567ospf_redistribute_set (struct ospf *ospf, int type, int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000568{
569 int force = 0;
paulcf795c52003-06-19 02:13:25 +0000570
paul718e3742002-12-13 20:15:29 +0000571 if (ospf_is_type_redistributed (type))
572 {
paul68980082003-03-25 05:07:42 +0000573 if (mtype != ospf->dmetric[type].type)
paulcf795c52003-06-19 02:13:25 +0000574 {
575 ospf->dmetric[type].type = mtype;
576 force = LSA_REFRESH_FORCE;
577 }
paul68980082003-03-25 05:07:42 +0000578 if (mvalue != ospf->dmetric[type].value)
paulcf795c52003-06-19 02:13:25 +0000579 {
580 ospf->dmetric[type].value = mvalue;
581 force = LSA_REFRESH_FORCE;
582 }
583
paul68980082003-03-25 05:07:42 +0000584 ospf_external_lsa_refresh_type (ospf, type, force);
paulcf795c52003-06-19 02:13:25 +0000585
paul718e3742002-12-13 20:15:29 +0000586 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000587 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000588 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000589 metric_type (ospf, type), metric_value (ospf, type));
590
paul718e3742002-12-13 20:15:29 +0000591 return CMD_SUCCESS;
592 }
593
paul68980082003-03-25 05:07:42 +0000594 ospf->dmetric[type].type = mtype;
595 ospf->dmetric[type].value = mvalue;
paul718e3742002-12-13 20:15:29 +0000596
paul0a589352004-05-08 11:48:26 +0000597 zclient_redistribute (ZEBRA_REDISTRIBUTE_ADD, zclient, type);
paul718e3742002-12-13 20:15:29 +0000598
599 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000600 zlog_debug ("Redistribute[%s]: Start Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000601 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000602 metric_type (ospf, type), metric_value (ospf, type));
603
paul68980082003-03-25 05:07:42 +0000604 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000605
606 return CMD_SUCCESS;
607}
608
609int
paul020709f2003-04-04 02:44:16 +0000610ospf_redistribute_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000611{
612 if (type == zclient->redist_default)
613 return CMD_SUCCESS;
614
paulcf795c52003-06-19 02:13:25 +0000615 if (!ospf_is_type_redistributed (type))
paul718e3742002-12-13 20:15:29 +0000616 return CMD_SUCCESS;
617
paul0a589352004-05-08 11:48:26 +0000618 zclient_redistribute (ZEBRA_REDISTRIBUTE_DELETE, zclient, type);
paulcf795c52003-06-19 02:13:25 +0000619
paul718e3742002-12-13 20:15:29 +0000620 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000621 zlog_debug ("Redistribute[%s]: Stop",
ajsf52d13c2005-10-01 17:38:06 +0000622 ospf_redist_string(type));
paul718e3742002-12-13 20:15:29 +0000623
paul68980082003-03-25 05:07:42 +0000624 ospf->dmetric[type].type = -1;
625 ospf->dmetric[type].value = -1;
paul718e3742002-12-13 20:15:29 +0000626
627 /* Remove the routes from OSPF table. */
Paul Jakma6db3a6f2006-05-12 23:02:46 +0000628 ospf_redistribute_withdraw (ospf, type);
paul718e3742002-12-13 20:15:29 +0000629
paul68980082003-03-25 05:07:42 +0000630 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000631
632 return CMD_SUCCESS;
633}
634
635int
paul020709f2003-04-04 02:44:16 +0000636ospf_redistribute_default_set (struct ospf *ospf, int originate,
paulcf795c52003-06-19 02:13:25 +0000637 int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000638{
Andrew J. Schorr8fb8a502006-10-24 19:04:26 +0000639 ospf->default_originate = originate;
640 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
641 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
paul020709f2003-04-04 02:44:16 +0000642
paul718e3742002-12-13 20:15:29 +0000643 if (ospf_is_type_redistributed (DEFAULT_ROUTE))
644 {
Andrew J. Schorr8fb8a502006-10-24 19:04:26 +0000645 /* if ospf->default_originate changes value, is calling
646 ospf_external_lsa_refresh_default sufficient to implement
647 the change? */
paul68980082003-03-25 05:07:42 +0000648 ospf_external_lsa_refresh_default (ospf);
paulcf795c52003-06-19 02:13:25 +0000649
paul718e3742002-12-13 20:15:29 +0000650 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000651 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000652 ospf_redist_string(DEFAULT_ROUTE),
paulcf795c52003-06-19 02:13:25 +0000653 metric_type (ospf, DEFAULT_ROUTE),
654 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000655 return CMD_SUCCESS;
656 }
657
paul0a589352004-05-08 11:48:26 +0000658 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_ADD, zclient);
paulcf795c52003-06-19 02:13:25 +0000659
paul718e3742002-12-13 20:15:29 +0000660 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000661 zlog_debug ("Redistribute[DEFAULT]: Start Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000662 metric_type (ospf, DEFAULT_ROUTE),
663 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000664
paul68980082003-03-25 05:07:42 +0000665 if (ospf->router_id.s_addr == 0)
666 ospf->external_origin |= (1 << DEFAULT_ROUTE);
paul718e3742002-12-13 20:15:29 +0000667 else
Paul Jakma4021b602006-05-12 22:55:41 +0000668 thread_add_timer (master, ospf_default_originate_timer, ospf, 1);
paul718e3742002-12-13 20:15:29 +0000669
paul68980082003-03-25 05:07:42 +0000670 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000671
672 return CMD_SUCCESS;
673}
674
675int
paul020709f2003-04-04 02:44:16 +0000676ospf_redistribute_default_unset (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +0000677{
678 if (!ospf_is_type_redistributed (DEFAULT_ROUTE))
679 return CMD_SUCCESS;
680
paul68980082003-03-25 05:07:42 +0000681 ospf->default_originate = DEFAULT_ORIGINATE_NONE;
682 ospf->dmetric[DEFAULT_ROUTE].type = -1;
683 ospf->dmetric[DEFAULT_ROUTE].value = -1;
paul718e3742002-12-13 20:15:29 +0000684
paul0a589352004-05-08 11:48:26 +0000685 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_DELETE, zclient);
paul718e3742002-12-13 20:15:29 +0000686
687 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000688 zlog_debug ("Redistribute[DEFAULT]: Stop");
paulcf795c52003-06-19 02:13:25 +0000689
paul68980082003-03-25 05:07:42 +0000690 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000691
692 return CMD_SUCCESS;
693}
694
paul4dadc292005-05-06 21:37:42 +0000695static int
paul020709f2003-04-04 02:44:16 +0000696ospf_external_lsa_originate_check (struct ospf *ospf,
paulcf795c52003-06-19 02:13:25 +0000697 struct external_info *ei)
paul718e3742002-12-13 20:15:29 +0000698{
699 /* If prefix is multicast, then do not originate LSA. */
700 if (IN_MULTICAST (htonl (ei->p.prefix.s_addr)))
701 {
702 zlog_info ("LSA[Type5:%s]: Not originate AS-external-LSA, "
paulcf795c52003-06-19 02:13:25 +0000703 "Prefix belongs multicast", inet_ntoa (ei->p.prefix));
paul718e3742002-12-13 20:15:29 +0000704 return 0;
705 }
706
707 /* Take care of default-originate. */
708 if (is_prefix_default (&ei->p))
paul68980082003-03-25 05:07:42 +0000709 if (ospf->default_originate == DEFAULT_ORIGINATE_NONE)
paul718e3742002-12-13 20:15:29 +0000710 {
Denis Ovsienkobcc6c592011-09-10 23:29:19 +0400711 zlog_info ("LSA[Type5:0.0.0.0]: Not originate AS-external-LSA "
paulcf795c52003-06-19 02:13:25 +0000712 "for default");
713 return 0;
paul718e3742002-12-13 20:15:29 +0000714 }
715
716 return 1;
717}
718
719/* If connected prefix is OSPF enable interface, then do not announce. */
720int
paulcf795c52003-06-19 02:13:25 +0000721ospf_distribute_check_connected (struct ospf *ospf, struct external_info *ei)
paul718e3742002-12-13 20:15:29 +0000722{
Joakim Tjernlund5d8de932009-08-07 13:48:15 +0200723 struct listnode *node;
724 struct ospf_interface *oi;
paul718e3742002-12-13 20:15:29 +0000725
Joakim Tjernlund5d8de932009-08-07 13:48:15 +0200726
727 for (ALL_LIST_ELEMENTS_RO (ospf->oiflist, node, oi))
728 if (prefix_match (oi->address, (struct prefix *) &ei->p))
paulcf795c52003-06-19 02:13:25 +0000729 return 0;
paul718e3742002-12-13 20:15:29 +0000730 return 1;
731}
732
733/* return 1 if external LSA must be originated, 0 otherwise */
734int
paul68980082003-03-25 05:07:42 +0000735ospf_redistribute_check (struct ospf *ospf,
paulcf795c52003-06-19 02:13:25 +0000736 struct external_info *ei, int *changed)
paul718e3742002-12-13 20:15:29 +0000737{
738 struct route_map_set_values save_values;
739 struct prefix_ipv4 *p = &ei->p;
740 u_char type = is_prefix_default (&ei->p) ? DEFAULT_ROUTE : ei->type;
paulcf795c52003-06-19 02:13:25 +0000741
paul718e3742002-12-13 20:15:29 +0000742 if (changed)
743 *changed = 0;
744
paul020709f2003-04-04 02:44:16 +0000745 if (!ospf_external_lsa_originate_check (ospf, ei))
paul718e3742002-12-13 20:15:29 +0000746 return 0;
747
748 /* Take care connected route. */
paul68980082003-03-25 05:07:42 +0000749 if (type == ZEBRA_ROUTE_CONNECT &&
750 !ospf_distribute_check_connected (ospf, ei))
paul718e3742002-12-13 20:15:29 +0000751 return 0;
752
paul020709f2003-04-04 02:44:16 +0000753 if (!DEFAULT_ROUTE_TYPE (type) && DISTRIBUTE_NAME (ospf, type))
paul718e3742002-12-13 20:15:29 +0000754 /* distirbute-list exists, but access-list may not? */
paul020709f2003-04-04 02:44:16 +0000755 if (DISTRIBUTE_LIST (ospf, type))
756 if (access_list_apply (DISTRIBUTE_LIST (ospf, type), p) == FILTER_DENY)
paulcf795c52003-06-19 02:13:25 +0000757 {
758 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000759 zlog_debug ("Redistribute[%s]: %s/%d filtered by ditribute-list.",
ajsf52d13c2005-10-01 17:38:06 +0000760 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000761 inet_ntoa (p->prefix), p->prefixlen);
762 return 0;
763 }
paul718e3742002-12-13 20:15:29 +0000764
765 save_values = ei->route_map_set;
766 ospf_reset_route_map_set_values (&ei->route_map_set);
paulcf795c52003-06-19 02:13:25 +0000767
paul718e3742002-12-13 20:15:29 +0000768 /* apply route-map if needed */
paul020709f2003-04-04 02:44:16 +0000769 if (ROUTEMAP_NAME (ospf, type))
paul718e3742002-12-13 20:15:29 +0000770 {
771 int ret;
772
paulcf795c52003-06-19 02:13:25 +0000773 ret = route_map_apply (ROUTEMAP (ospf, type), (struct prefix *) p,
774 RMAP_OSPF, ei);
paul718e3742002-12-13 20:15:29 +0000775
776 if (ret == RMAP_DENYMATCH)
paulcf795c52003-06-19 02:13:25 +0000777 {
778 ei->route_map_set = save_values;
779 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000780 zlog_debug ("Redistribute[%s]: %s/%d filtered by route-map.",
ajsf52d13c2005-10-01 17:38:06 +0000781 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000782 inet_ntoa (p->prefix), p->prefixlen);
783 return 0;
784 }
785
paul718e3742002-12-13 20:15:29 +0000786 /* check if 'route-map set' changed something */
787 if (changed)
paulcf795c52003-06-19 02:13:25 +0000788 *changed = !ospf_route_map_set_compare (&ei->route_map_set,
789 &save_values);
paul718e3742002-12-13 20:15:29 +0000790 }
791
792 return 1;
793}
794
795/* OSPF route-map set for redistribution */
796void
paul6c835672004-10-11 11:00:30 +0000797ospf_routemap_set (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000798{
paul020709f2003-04-04 02:44:16 +0000799 if (ROUTEMAP_NAME (ospf, type))
800 free (ROUTEMAP_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000801
paul020709f2003-04-04 02:44:16 +0000802 ROUTEMAP_NAME (ospf, type) = strdup (name);
803 ROUTEMAP (ospf, type) = route_map_lookup_by_name (name);
paul718e3742002-12-13 20:15:29 +0000804}
805
806void
paul020709f2003-04-04 02:44:16 +0000807ospf_routemap_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000808{
paul020709f2003-04-04 02:44:16 +0000809 if (ROUTEMAP_NAME (ospf, type))
810 free (ROUTEMAP_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000811
paul020709f2003-04-04 02:44:16 +0000812 ROUTEMAP_NAME (ospf, type) = NULL;
813 ROUTEMAP (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000814}
815
816/* Zebra route add and delete treatment. */
paul4dadc292005-05-06 21:37:42 +0000817static int
paul718e3742002-12-13 20:15:29 +0000818ospf_zebra_read_ipv4 (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000819 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000820{
821 struct stream *s;
822 struct zapi_ipv4 api;
823 unsigned long ifindex;
824 struct in_addr nexthop;
825 struct prefix_ipv4 p;
826 struct external_info *ei;
paul020709f2003-04-04 02:44:16 +0000827 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000828
829 s = zclient->ibuf;
830 ifindex = 0;
831 nexthop.s_addr = 0;
832
833 /* Type, flags, message. */
834 api.type = stream_getc (s);
835 api.flags = stream_getc (s);
836 api.message = stream_getc (s);
837
838 /* IPv4 prefix. */
839 memset (&p, 0, sizeof (struct prefix_ipv4));
840 p.family = AF_INET;
841 p.prefixlen = stream_getc (s);
842 stream_get (&p.prefix, s, PSIZE (p.prefixlen));
843
hasso8585d4e2004-04-20 17:25:12 +0000844 if (IPV4_NET127(ntohl(p.prefix.s_addr)))
845 return 0;
846
paul718e3742002-12-13 20:15:29 +0000847 /* Nexthop, ifindex, distance, metric. */
848 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP))
849 {
850 api.nexthop_num = stream_getc (s);
851 nexthop.s_addr = stream_get_ipv4 (s);
852 }
853 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_IFINDEX))
854 {
855 api.ifindex_num = stream_getc (s);
gdt6a8da852004-02-13 17:40:51 +0000856 /* XXX assert(api.ifindex_num == 1); */
paul718e3742002-12-13 20:15:29 +0000857 ifindex = stream_getl (s);
858 }
859 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_DISTANCE))
860 api.distance = stream_getc (s);
861 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_METRIC))
862 api.metric = stream_getl (s);
863
paul020709f2003-04-04 02:44:16 +0000864 ospf = ospf_lookup ();
865 if (ospf == NULL)
866 return 0;
867
paul718e3742002-12-13 20:15:29 +0000868 if (command == ZEBRA_IPV4_ROUTE_ADD)
869 {
paul7021c422003-07-15 12:52:22 +0000870 /* XXX|HACK|TODO|FIXME:
hassoa0a39762004-04-23 08:51:10 +0000871 * Maybe we should ignore reject/blackhole routes? Testing shows that
872 * there is no problems though and this is only way to "summarize"
873 * routes in ASBR at the moment. Maybe we need just a better generalised
874 * solution for these types?
875 *
876 * if ( CHECK_FLAG (api.flags, ZEBRA_FLAG_BLACKHOLE)
877 * || CHECK_FLAG (api.flags, ZEBRA_FLAG_REJECT))
878 * return 0;
paul7021c422003-07-15 12:52:22 +0000879 */
paul7021c422003-07-15 12:52:22 +0000880
paul718e3742002-12-13 20:15:29 +0000881 ei = ospf_external_info_add (api.type, p, ifindex, nexthop);
882
paul68980082003-03-25 05:07:42 +0000883 if (ospf->router_id.s_addr == 0)
paulcf795c52003-06-19 02:13:25 +0000884 /* Set flags to generate AS-external-LSA originate event
885 for each redistributed protocols later. */
886 ospf->external_origin |= (1 << api.type);
paul718e3742002-12-13 20:15:29 +0000887 else
paulcf795c52003-06-19 02:13:25 +0000888 {
889 if (ei)
890 {
891 if (is_prefix_default (&p))
892 ospf_external_lsa_refresh_default (ospf);
893 else
894 {
895 struct ospf_lsa *current;
paul718e3742002-12-13 20:15:29 +0000896
paulcf795c52003-06-19 02:13:25 +0000897 current = ospf_external_info_find_lsa (ospf, &ei->p);
898 if (!current)
899 ospf_external_lsa_originate (ospf, ei);
900 else if (IS_LSA_MAXAGE (current))
901 ospf_external_lsa_refresh (ospf, current,
902 ei, LSA_REFRESH_FORCE);
903 else
904 zlog_warn ("ospf_zebra_read_ipv4() : %s already exists",
905 inet_ntoa (p.prefix));
906 }
907 }
908 }
paul718e3742002-12-13 20:15:29 +0000909 }
paulcf795c52003-06-19 02:13:25 +0000910 else /* if (command == ZEBRA_IPV4_ROUTE_DELETE) */
paul718e3742002-12-13 20:15:29 +0000911 {
912 ospf_external_info_delete (api.type, p);
paul7021c422003-07-15 12:52:22 +0000913 if (is_prefix_default (&p))
paulcf795c52003-06-19 02:13:25 +0000914 ospf_external_lsa_refresh_default (ospf);
paul7021c422003-07-15 12:52:22 +0000915 else
ajs5339cfd2005-09-19 13:28:05 +0000916 ospf_external_lsa_flush (ospf, api.type, &p, ifindex /*, nexthop */);
paul718e3742002-12-13 20:15:29 +0000917 }
918
919 return 0;
920}
paul718e3742002-12-13 20:15:29 +0000921
paulcf795c52003-06-19 02:13:25 +0000922
paul718e3742002-12-13 20:15:29 +0000923int
paul6c835672004-10-11 11:00:30 +0000924ospf_distribute_list_out_set (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000925{
926 /* Lookup access-list for distribute-list. */
paul020709f2003-04-04 02:44:16 +0000927 DISTRIBUTE_LIST (ospf, type) = access_list_lookup (AFI_IP, name);
paul718e3742002-12-13 20:15:29 +0000928
929 /* Clear previous distribute-name. */
paul020709f2003-04-04 02:44:16 +0000930 if (DISTRIBUTE_NAME (ospf, type))
931 free (DISTRIBUTE_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000932
933 /* Set distribute-name. */
paul020709f2003-04-04 02:44:16 +0000934 DISTRIBUTE_NAME (ospf, type) = strdup (name);
paul718e3742002-12-13 20:15:29 +0000935
936 /* If access-list have been set, schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000937 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000938 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000939
940 return CMD_SUCCESS;
941}
942
943int
paul6c835672004-10-11 11:00:30 +0000944ospf_distribute_list_out_unset (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000945{
946 /* Schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000947 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000948 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000949
950 /* Unset distribute-list. */
paul020709f2003-04-04 02:44:16 +0000951 DISTRIBUTE_LIST (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000952
953 /* Clear distribute-name. */
paul020709f2003-04-04 02:44:16 +0000954 if (DISTRIBUTE_NAME (ospf, type))
955 free (DISTRIBUTE_NAME (ospf, type));
paulcf795c52003-06-19 02:13:25 +0000956
paul020709f2003-04-04 02:44:16 +0000957 DISTRIBUTE_NAME (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000958
959 return CMD_SUCCESS;
960}
961
962/* distribute-list update timer. */
paul4dadc292005-05-06 21:37:42 +0000963static int
paul718e3742002-12-13 20:15:29 +0000964ospf_distribute_list_update_timer (struct thread *thread)
965{
966 struct route_node *rn;
967 struct external_info *ei;
968 struct route_table *rt;
969 struct ospf_lsa *lsa;
Joakim Tjernlund46154fe2010-04-14 16:01:25 +0200970 int type, default_refresh = 0;
paul020709f2003-04-04 02:44:16 +0000971 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000972
paul020709f2003-04-04 02:44:16 +0000973 ospf = ospf_lookup ();
974 if (ospf == NULL)
975 return 0;
976
paul68980082003-03-25 05:07:42 +0000977 ospf->t_distribute_update = NULL;
paul718e3742002-12-13 20:15:29 +0000978
979 zlog_info ("Zebra[Redistribute]: distribute-list update timer fired!");
980
981 /* foreach all external info. */
Joakim Tjernlund274d3f02010-04-14 11:05:27 +0200982 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
983 {
984 rt = EXTERNAL_INFO (type);
985 if (!rt)
986 continue;
987 for (rn = route_top (rt); rn; rn = route_next (rn))
988 if ((ei = rn->info) != NULL)
989 {
990 if (is_prefix_default (&ei->p))
Joakim Tjernlund46154fe2010-04-14 16:01:25 +0200991 default_refresh = 1;
Joakim Tjernlund274d3f02010-04-14 11:05:27 +0200992 else if ((lsa = ospf_external_info_find_lsa (ospf, &ei->p)))
993 ospf_external_lsa_refresh (ospf, lsa, ei, LSA_REFRESH_IF_CHANGED);
994 else
995 ospf_external_lsa_originate (ospf, ei);
996 }
997 }
Joakim Tjernlund46154fe2010-04-14 16:01:25 +0200998 if (default_refresh)
999 ospf_external_lsa_refresh_default (ospf);
paul718e3742002-12-13 20:15:29 +00001000 return 0;
1001}
1002
1003#define OSPF_DISTRIBUTE_UPDATE_DELAY 5
1004
1005/* Update distribute-list and set timer to apply access-list. */
1006void
paul68980082003-03-25 05:07:42 +00001007ospf_distribute_list_update (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +00001008{
1009 struct route_table *rt;
paulcf795c52003-06-19 02:13:25 +00001010
paul718e3742002-12-13 20:15:29 +00001011 /* External info does not exist. */
1012 if (!(rt = EXTERNAL_INFO (type)))
1013 return;
1014
Joakim Tjernlund45acaa02010-04-14 11:05:28 +02001015 /* If exists previously invoked thread, then let it continue. */
paul68980082003-03-25 05:07:42 +00001016 if (ospf->t_distribute_update)
Joakim Tjernlund45acaa02010-04-14 11:05:28 +02001017 return;
paul718e3742002-12-13 20:15:29 +00001018
1019 /* Set timer. */
paul68980082003-03-25 05:07:42 +00001020 ospf->t_distribute_update =
paul718e3742002-12-13 20:15:29 +00001021 thread_add_timer (master, ospf_distribute_list_update_timer,
paulcf795c52003-06-19 02:13:25 +00001022 (void *) type, OSPF_DISTRIBUTE_UPDATE_DELAY);
paul718e3742002-12-13 20:15:29 +00001023}
1024
1025/* If access-list is updated, apply some check. */
paul4dadc292005-05-06 21:37:42 +00001026static void
paul718e3742002-12-13 20:15:29 +00001027ospf_filter_update (struct access_list *access)
1028{
paul020709f2003-04-04 02:44:16 +00001029 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +00001030 int type;
1031 int abr_inv = 0;
1032 struct ospf_area *area;
hasso52dc7ee2004-09-23 19:18:23 +00001033 struct listnode *node;
paul718e3742002-12-13 20:15:29 +00001034
1035 /* If OSPF instatnce does not exist, return right now. */
paul020709f2003-04-04 02:44:16 +00001036 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +00001037 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +00001038 return;
1039
paul718e3742002-12-13 20:15:29 +00001040 /* Update distribute-list, and apply filter. */
hasso01018ce2005-08-05 07:40:15 +00001041 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
paul718e3742002-12-13 20:15:29 +00001042 {
paul020709f2003-04-04 02:44:16 +00001043 if (ROUTEMAP (ospf, type) != NULL)
paulcf795c52003-06-19 02:13:25 +00001044 {
1045 /* if route-map is not NULL it may be using this access list */
1046 ospf_distribute_list_update (ospf, type);
1047 continue;
1048 }
1049
hasso01018ce2005-08-05 07:40:15 +00001050 /* There is place for route-map for default-information (ZEBRA_ROUTE_MAX),
1051 * but no distribute list. */
1052 if (type == ZEBRA_ROUTE_MAX)
1053 break;
paul718e3742002-12-13 20:15:29 +00001054
paul020709f2003-04-04 02:44:16 +00001055 if (DISTRIBUTE_NAME (ospf, type))
paulcf795c52003-06-19 02:13:25 +00001056 {
1057 /* Keep old access-list for distribute-list. */
1058 struct access_list *old = DISTRIBUTE_LIST (ospf, type);
1059
1060 /* Update access-list for distribute-list. */
1061 DISTRIBUTE_LIST (ospf, type) =
1062 access_list_lookup (AFI_IP, DISTRIBUTE_NAME (ospf, type));
1063
1064 /* No update for this distribute type. */
1065 if (old == NULL && DISTRIBUTE_LIST (ospf, type) == NULL)
1066 continue;
1067
1068 /* Schedule distribute-list update timer. */
1069 if (DISTRIBUTE_LIST (ospf, type) == NULL ||
1070 strcmp (DISTRIBUTE_NAME (ospf, type), access->name) == 0)
1071 ospf_distribute_list_update (ospf, type);
1072 }
paul718e3742002-12-13 20:15:29 +00001073 }
1074
1075 /* Update Area access-list. */
paul1eb8ef22005-04-07 07:30:20 +00001076 for (ALL_LIST_ELEMENTS_RO (ospf->areas, node, area))
1077 {
1078 if (EXPORT_NAME (area))
1079 {
1080 EXPORT_LIST (area) = NULL;
1081 abr_inv++;
1082 }
paul718e3742002-12-13 20:15:29 +00001083
paul1eb8ef22005-04-07 07:30:20 +00001084 if (IMPORT_NAME (area))
1085 {
1086 IMPORT_LIST (area) = NULL;
1087 abr_inv++;
1088 }
1089 }
paul718e3742002-12-13 20:15:29 +00001090
1091 /* Schedule ABR tasks -- this will be changed -- takada. */
paul020709f2003-04-04 02:44:16 +00001092 if (IS_OSPF_ABR (ospf) && abr_inv)
paul68980082003-03-25 05:07:42 +00001093 ospf_schedule_abr_task (ospf);
paul718e3742002-12-13 20:15:29 +00001094}
hassodd669bb2004-05-10 07:43:59 +00001095
1096/* If prefix-list is updated, do some updates. */
1097void
1098ospf_prefix_list_update (struct prefix_list *plist)
1099{
1100 struct ospf *ospf;
1101 int type;
1102 int abr_inv = 0;
1103 struct ospf_area *area;
hasso52dc7ee2004-09-23 19:18:23 +00001104 struct listnode *node;
hassodd669bb2004-05-10 07:43:59 +00001105
1106 /* If OSPF instatnce does not exist, return right now. */
1107 ospf = ospf_lookup ();
1108 if (ospf == NULL)
1109 return;
1110
1111 /* Update all route-maps which are used as redistribution filters.
1112 * They might use prefix-list.
1113 */
hasso01018ce2005-08-05 07:40:15 +00001114 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
hassodd669bb2004-05-10 07:43:59 +00001115 {
1116 if (ROUTEMAP (ospf, type) != NULL)
1117 {
1118 /* If route-map is not NULL it may be using this prefix list */
1119 ospf_distribute_list_update (ospf, type);
1120 continue;
1121 }
1122 }
1123
1124 /* Update area filter-lists. */
paul1eb8ef22005-04-07 07:30:20 +00001125 for (ALL_LIST_ELEMENTS_RO (ospf->areas, node, area))
1126 {
1127 /* Update filter-list in. */
1128 if (PREFIX_NAME_IN (area))
1129 if (strcmp (PREFIX_NAME_IN (area), plist->name) == 0)
1130 {
1131 PREFIX_LIST_IN (area) =
1132 prefix_list_lookup (AFI_IP, PREFIX_NAME_IN (area));
1133 abr_inv++;
1134 }
hassodd669bb2004-05-10 07:43:59 +00001135
paul1eb8ef22005-04-07 07:30:20 +00001136 /* Update filter-list out. */
1137 if (PREFIX_NAME_OUT (area))
1138 if (strcmp (PREFIX_NAME_OUT (area), plist->name) == 0)
1139 {
1140 PREFIX_LIST_IN (area) =
1141 prefix_list_lookup (AFI_IP, PREFIX_NAME_OUT (area));
1142 abr_inv++;
1143 }
1144 }
hassodd669bb2004-05-10 07:43:59 +00001145
1146 /* Schedule ABR task. */
1147 if (IS_OSPF_ABR (ospf) && abr_inv)
1148 ospf_schedule_abr_task (ospf);
1149}
paulcf795c52003-06-19 02:13:25 +00001150
paul4dadc292005-05-06 21:37:42 +00001151static struct ospf_distance *
1152ospf_distance_new (void)
paul718e3742002-12-13 20:15:29 +00001153{
Stephen Hemminger393deb92008-08-18 14:13:29 -07001154 return XCALLOC (MTYPE_OSPF_DISTANCE, sizeof (struct ospf_distance));
paul718e3742002-12-13 20:15:29 +00001155}
1156
paul4dadc292005-05-06 21:37:42 +00001157static void
paul718e3742002-12-13 20:15:29 +00001158ospf_distance_free (struct ospf_distance *odistance)
1159{
1160 XFREE (MTYPE_OSPF_DISTANCE, odistance);
1161}
1162
1163int
paul6c835672004-10-11 11:00:30 +00001164ospf_distance_set (struct vty *vty, struct ospf *ospf,
1165 const char *distance_str,
1166 const char *ip_str,
1167 const char *access_list_str)
paul718e3742002-12-13 20:15:29 +00001168{
1169 int ret;
1170 struct prefix_ipv4 p;
1171 u_char distance;
1172 struct route_node *rn;
1173 struct ospf_distance *odistance;
1174
1175 ret = str2prefix_ipv4 (ip_str, &p);
1176 if (ret == 0)
1177 {
1178 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1179 return CMD_WARNING;
1180 }
1181
1182 distance = atoi (distance_str);
1183
1184 /* Get OSPF distance node. */
paul68980082003-03-25 05:07:42 +00001185 rn = route_node_get (ospf->distance_table, (struct prefix *) &p);
paul718e3742002-12-13 20:15:29 +00001186 if (rn->info)
1187 {
1188 odistance = rn->info;
1189 route_unlock_node (rn);
1190 }
1191 else
1192 {
1193 odistance = ospf_distance_new ();
1194 rn->info = odistance;
1195 }
1196
1197 /* Set distance value. */
1198 odistance->distance = distance;
1199
1200 /* Reset access-list configuration. */
1201 if (odistance->access_list)
1202 {
1203 free (odistance->access_list);
1204 odistance->access_list = NULL;
1205 }
1206 if (access_list_str)
1207 odistance->access_list = strdup (access_list_str);
1208
1209 return CMD_SUCCESS;
1210}
1211
1212int
paul6c835672004-10-11 11:00:30 +00001213ospf_distance_unset (struct vty *vty, struct ospf *ospf,
1214 const char *distance_str,
1215 const char *ip_str, char
1216 const *access_list_str)
paul718e3742002-12-13 20:15:29 +00001217{
1218 int ret;
1219 struct prefix_ipv4 p;
1220 u_char distance;
1221 struct route_node *rn;
1222 struct ospf_distance *odistance;
1223
1224 ret = str2prefix_ipv4 (ip_str, &p);
1225 if (ret == 0)
1226 {
1227 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1228 return CMD_WARNING;
1229 }
1230
1231 distance = atoi (distance_str);
1232
paulcf795c52003-06-19 02:13:25 +00001233 rn = route_node_lookup (ospf->distance_table, (struct prefix *) &p);
1234 if (!rn)
paul718e3742002-12-13 20:15:29 +00001235 {
1236 vty_out (vty, "Can't find specified prefix%s", VTY_NEWLINE);
1237 return CMD_WARNING;
1238 }
1239
1240 odistance = rn->info;
1241
1242 if (odistance->access_list)
1243 free (odistance->access_list);
1244 ospf_distance_free (odistance);
1245
1246 rn->info = NULL;
1247 route_unlock_node (rn);
1248 route_unlock_node (rn);
1249
1250 return CMD_SUCCESS;
1251}
1252
1253void
paul68980082003-03-25 05:07:42 +00001254ospf_distance_reset (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +00001255{
1256 struct route_node *rn;
1257 struct ospf_distance *odistance;
1258
paul68980082003-03-25 05:07:42 +00001259 for (rn = route_top (ospf->distance_table); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +00001260 if ((odistance = rn->info) != NULL)
1261 {
paulcf795c52003-06-19 02:13:25 +00001262 if (odistance->access_list)
1263 free (odistance->access_list);
1264 ospf_distance_free (odistance);
1265 rn->info = NULL;
1266 route_unlock_node (rn);
paul718e3742002-12-13 20:15:29 +00001267 }
1268}
1269
1270u_char
1271ospf_distance_apply (struct prefix_ipv4 *p, struct ospf_route *or)
1272{
paul020709f2003-04-04 02:44:16 +00001273 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +00001274
paul020709f2003-04-04 02:44:16 +00001275 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +00001276 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +00001277 return 0;
1278
paul68980082003-03-25 05:07:42 +00001279 if (ospf->distance_intra)
paul718e3742002-12-13 20:15:29 +00001280 if (or->path_type == OSPF_PATH_INTRA_AREA)
paul68980082003-03-25 05:07:42 +00001281 return ospf->distance_intra;
paul718e3742002-12-13 20:15:29 +00001282
paul68980082003-03-25 05:07:42 +00001283 if (ospf->distance_inter)
paul718e3742002-12-13 20:15:29 +00001284 if (or->path_type == OSPF_PATH_INTER_AREA)
paul68980082003-03-25 05:07:42 +00001285 return ospf->distance_inter;
paul718e3742002-12-13 20:15:29 +00001286
paul68980082003-03-25 05:07:42 +00001287 if (ospf->distance_external)
paul718e3742002-12-13 20:15:29 +00001288 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL
paulcf795c52003-06-19 02:13:25 +00001289 || or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
paul68980082003-03-25 05:07:42 +00001290 return ospf->distance_external;
paulcf795c52003-06-19 02:13:25 +00001291
paul68980082003-03-25 05:07:42 +00001292 if (ospf->distance_all)
1293 return ospf->distance_all;
paul718e3742002-12-13 20:15:29 +00001294
1295 return 0;
1296}
1297
1298void
1299ospf_zebra_init ()
1300{
1301 /* Allocate zebra structure. */
1302 zclient = zclient_new ();
1303 zclient_init (zclient, ZEBRA_ROUTE_OSPF);
hasso18a6dce2004-10-03 18:18:34 +00001304 zclient->router_id_update = ospf_router_id_update_zebra;
paul718e3742002-12-13 20:15:29 +00001305 zclient->interface_add = ospf_interface_add;
1306 zclient->interface_delete = ospf_interface_delete;
1307 zclient->interface_up = ospf_interface_state_up;
1308 zclient->interface_down = ospf_interface_state_down;
1309 zclient->interface_address_add = ospf_interface_address_add;
1310 zclient->interface_address_delete = ospf_interface_address_delete;
1311 zclient->ipv4_route_add = ospf_zebra_read_ipv4;
1312 zclient->ipv4_route_delete = ospf_zebra_read_ipv4;
1313
1314 access_list_add_hook (ospf_filter_update);
1315 access_list_delete_hook (ospf_filter_update);
hassodd669bb2004-05-10 07:43:59 +00001316 prefix_list_add_hook (ospf_prefix_list_update);
1317 prefix_list_delete_hook (ospf_prefix_list_update);
paul718e3742002-12-13 20:15:29 +00001318}