blob: bdd01424cf32faa8bbc8de30f79674bd7ec98efd [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;
paul020709f2003-04-04 02:44:16 +000090 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +000091
92 ifp = zebra_interface_add_read (zclient->ibuf);
93
94 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +000095 zlog_debug ("Zebra: interface add %s index %d flags %ld metric %d mtu %d",
paulcf795c52003-06-19 02:13:25 +000096 ifp->name, ifp->ifindex, ifp->flags, 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
paul020709f2003-04-04 02:44:16 +0000106 ospf = ospf_lookup ();
107 if (ospf != NULL)
108 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000109
110#ifdef HAVE_SNMP
111 ospf_snmp_if_update (ifp);
112#endif /* HAVE_SNMP */
113
114 return 0;
115}
116
paul4dadc292005-05-06 21:37:42 +0000117static int
paul718e3742002-12-13 20:15:29 +0000118ospf_interface_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000119 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000120{
121 struct interface *ifp;
122 struct stream *s;
123 struct route_node *rn;
124
paulcf795c52003-06-19 02:13:25 +0000125 s = zclient->ibuf;
paul718e3742002-12-13 20:15:29 +0000126 /* zebra_interface_state_read() updates interface structure in iflist */
127 ifp = zebra_interface_state_read (s);
128
129 if (ifp == NULL)
130 return 0;
131
132 if (if_is_up (ifp))
133 zlog_warn ("Zebra: got delete of %s, but interface is still up",
paulcf795c52003-06-19 02:13:25 +0000134 ifp->name);
135
paul718e3742002-12-13 20:15:29 +0000136 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000137 zlog_debug
paulcf795c52003-06-19 02:13:25 +0000138 ("Zebra: interface delete %s index %d flags %ld metric %d mtu %d",
139 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +0000140
141#ifdef HAVE_SNMP
142 ospf_snmp_if_delete (ifp);
143#endif /* HAVE_SNMP */
144
145 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
146 if (rn->info)
147 ospf_if_free ((struct ospf_interface *) rn->info);
148
ajsd2fc8892005-04-02 18:38:43 +0000149 ifp->ifindex = IFINDEX_INTERNAL;
paul718e3742002-12-13 20:15:29 +0000150 return 0;
151}
152
ajsd2fc8892005-04-02 18:38:43 +0000153static struct interface *
paul718e3742002-12-13 20:15:29 +0000154zebra_interface_if_lookup (struct stream *s)
155{
ajs21fefa92005-04-02 23:16:41 +0000156 char ifname_tmp[INTERFACE_NAMSIZ];
paul718e3742002-12-13 20:15:29 +0000157
158 /* Read interface name. */
159 stream_get (ifname_tmp, s, INTERFACE_NAMSIZ);
160
ajsd2fc8892005-04-02 18:38:43 +0000161 /* And look it up. */
ajs21fefa92005-04-02 23:16:41 +0000162 return if_lookup_by_name_len(ifname_tmp,
163 strnlen(ifname_tmp, INTERFACE_NAMSIZ));
paul718e3742002-12-13 20:15:29 +0000164}
165
paul4dadc292005-05-06 21:37:42 +0000166static int
paul718e3742002-12-13 20:15:29 +0000167ospf_interface_state_up (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000168 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000169{
170 struct interface *ifp;
paul718e3742002-12-13 20:15:29 +0000171 struct ospf_interface *oi;
172 struct route_node *rn;
paulcf795c52003-06-19 02:13:25 +0000173
paul718e3742002-12-13 20:15:29 +0000174 ifp = zebra_interface_if_lookup (zclient->ibuf);
175
176 if (ifp == NULL)
177 return 0;
178
179 /* Interface is already up. */
paul2e3b2e42002-12-13 21:03:13 +0000180 if (if_is_operative (ifp))
paul718e3742002-12-13 20:15:29 +0000181 {
182 /* Temporarily keep ifp values. */
ajsa608bbf2005-03-29 17:03:49 +0000183 struct interface if_tmp;
paul718e3742002-12-13 20:15:29 +0000184 memcpy (&if_tmp, ifp, sizeof (struct interface));
185
186 zebra_interface_if_set_value (zclient->ibuf, ifp);
187
188 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000189 zlog_debug ("Zebra: Interface[%s] state update.", ifp->name);
paul718e3742002-12-13 20:15:29 +0000190
191 if (if_tmp.bandwidth != ifp->bandwidth)
paulcf795c52003-06-19 02:13:25 +0000192 {
193 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000194 zlog_debug ("Zebra: Interface[%s] bandwidth change %d -> %d.",
paulcf795c52003-06-19 02:13:25 +0000195 ifp->name, if_tmp.bandwidth, ifp->bandwidth);
paul718e3742002-12-13 20:15:29 +0000196
paulcf795c52003-06-19 02:13:25 +0000197 ospf_if_recalculate_output_cost (ifp);
198 }
ajsa608bbf2005-03-29 17:03:49 +0000199
200 if (if_tmp.mtu != ifp->mtu)
201 {
202 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
203 zlog_debug ("Zebra: Interface[%s] MTU change %u -> %u.",
204 ifp->name, if_tmp.mtu, ifp->mtu);
205
206 /* Must reset the interface (simulate down/up) when MTU changes. */
207 ospf_if_reset(ifp);
208 }
paul718e3742002-12-13 20:15:29 +0000209 return 0;
210 }
paulcf795c52003-06-19 02:13:25 +0000211
paul718e3742002-12-13 20:15:29 +0000212 zebra_interface_if_set_value (zclient->ibuf, ifp);
paulcf795c52003-06-19 02:13:25 +0000213
paul718e3742002-12-13 20:15:29 +0000214 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000215 zlog_debug ("Zebra: Interface[%s] state change to up.", ifp->name);
paulcf795c52003-06-19 02:13:25 +0000216
217 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +0000218 {
paulcf795c52003-06-19 02:13:25 +0000219 if ((oi = rn->info) == NULL)
220 continue;
221
paul718e3742002-12-13 20:15:29 +0000222 ospf_if_up (oi);
223 }
paulcf795c52003-06-19 02:13:25 +0000224
paul718e3742002-12-13 20:15:29 +0000225 return 0;
226}
227
paul4dadc292005-05-06 21:37:42 +0000228static int
paul718e3742002-12-13 20:15:29 +0000229ospf_interface_state_down (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000230 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000231{
232 struct interface *ifp;
233 struct ospf_interface *oi;
234 struct route_node *node;
235
236 ifp = zebra_interface_state_read (zclient->ibuf);
237
238 if (ifp == NULL)
239 return 0;
240
241 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000242 zlog_debug ("Zebra: Interface[%s] state change to down.", ifp->name);
paul718e3742002-12-13 20:15:29 +0000243
paulcf795c52003-06-19 02:13:25 +0000244 for (node = route_top (IF_OIFS (ifp)); node; node = route_next (node))
paul718e3742002-12-13 20:15:29 +0000245 {
paulcf795c52003-06-19 02:13:25 +0000246 if ((oi = node->info) == NULL)
247 continue;
paul718e3742002-12-13 20:15:29 +0000248 ospf_if_down (oi);
249 }
250
251 return 0;
252}
253
paul4dadc292005-05-06 21:37:42 +0000254static int
paul718e3742002-12-13 20:15:29 +0000255ospf_interface_address_add (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000256 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000257{
paul020709f2003-04-04 02:44:16 +0000258 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000259 struct connected *c;
260
paul0a589352004-05-08 11:48:26 +0000261 c = zebra_interface_address_read (command, zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000262
263 if (c == NULL)
264 return 0;
265
Andrew J. Schorr7f643eb2006-11-30 16:17:02 +0000266 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
267 {
268 char buf[128];
269 prefix2str(c->address, buf, sizeof(buf));
270 zlog_debug("Zebra: interface %s address add %s", c->ifp->name, buf);
271 }
272
paul020709f2003-04-04 02:44:16 +0000273 ospf = ospf_lookup ();
274 if (ospf != NULL)
275 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000276
277#ifdef HAVE_SNMP
278 ospf_snmp_if_update (c->ifp);
279#endif /* HAVE_SNMP */
280
281 return 0;
282}
283
paul4dadc292005-05-06 21:37:42 +0000284static int
paul718e3742002-12-13 20:15:29 +0000285ospf_interface_address_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000286 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000287{
paul020709f2003-04-04 02:44:16 +0000288 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000289 struct connected *c;
290 struct interface *ifp;
291 struct ospf_interface *oi;
292 struct route_node *rn;
293 struct prefix p;
294
paul0a589352004-05-08 11:48:26 +0000295 c = zebra_interface_address_read (command, zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000296
297 if (c == NULL)
298 return 0;
299
Andrew J. Schorr7f643eb2006-11-30 16:17:02 +0000300 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
301 {
302 char buf[128];
303 prefix2str(c->address, buf, sizeof(buf));
304 zlog_debug("Zebra: interface %s address delete %s", c->ifp->name, buf);
305 }
306
paul718e3742002-12-13 20:15:29 +0000307 ifp = c->ifp;
308 p = *c->address;
309 p.prefixlen = IPV4_MAX_PREFIXLEN;
310
311 rn = route_node_lookup (IF_OIFS (ifp), &p);
paulcf795c52003-06-19 02:13:25 +0000312 if (!rn)
paul98429f62006-01-10 22:11:54 +0000313 {
314 connected_free (c);
315 return 0;
316 }
paul718e3742002-12-13 20:15:29 +0000317
318 assert (rn->info);
319 oi = rn->info;
paulcf795c52003-06-19 02:13:25 +0000320
paul718e3742002-12-13 20:15:29 +0000321 /* Call interface hook functions to clean up */
322 ospf_if_free (oi);
paulcf795c52003-06-19 02:13:25 +0000323
paul718e3742002-12-13 20:15:29 +0000324#ifdef HAVE_SNMP
325 ospf_snmp_if_update (c->ifp);
326#endif /* HAVE_SNMP */
327
328 connected_free (c);
329
paul020709f2003-04-04 02:44:16 +0000330 ospf = ospf_lookup ();
331 if (ospf != NULL)
332 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000333
334 return 0;
335}
paul72357f22003-06-19 02:11:23 +0000336
paul718e3742002-12-13 20:15:29 +0000337void
338ospf_zebra_add (struct prefix_ipv4 *p, struct ospf_route *or)
339{
340 u_char message;
341 u_char distance;
342 u_char flags;
343 int psize;
344 struct stream *s;
345 struct ospf_path *path;
hasso52dc7ee2004-09-23 19:18:23 +0000346 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000347
348 if (zclient->redist[ZEBRA_ROUTE_OSPF])
349 {
350 message = 0;
351 flags = 0;
352
353 /* OSPF pass nexthop and metric */
354 SET_FLAG (message, ZAPI_MESSAGE_NEXTHOP);
355 SET_FLAG (message, ZAPI_MESSAGE_METRIC);
356
357 /* Distance value. */
358 distance = ospf_distance_apply (p, or);
359 if (distance)
paul72357f22003-06-19 02:11:23 +0000360 SET_FLAG (message, ZAPI_MESSAGE_DISTANCE);
paul718e3742002-12-13 20:15:29 +0000361
362 /* Make packet. */
363 s = zclient->obuf;
364 stream_reset (s);
365
paul718e3742002-12-13 20:15:29 +0000366 /* Put command, type, flags, message. */
paulc6371712006-01-17 17:49:53 +0000367 zclient_create_header (s, ZEBRA_IPV4_ROUTE_ADD);
paul718e3742002-12-13 20:15:29 +0000368 stream_putc (s, ZEBRA_ROUTE_OSPF);
369 stream_putc (s, flags);
370 stream_putc (s, message);
paulcf795c52003-06-19 02:13:25 +0000371
paul718e3742002-12-13 20:15:29 +0000372 /* Put prefix information. */
373 psize = PSIZE (p->prefixlen);
374 stream_putc (s, p->prefixlen);
paulcf795c52003-06-19 02:13:25 +0000375 stream_write (s, (u_char *) & p->prefix, psize);
paul718e3742002-12-13 20:15:29 +0000376
377 /* Nexthop count. */
paul96735ee2003-08-10 02:51:22 +0000378 stream_putc (s, or->paths->count);
paul718e3742002-12-13 20:15:29 +0000379
380 /* Nexthop, ifindex, distance and metric information. */
paul1eb8ef22005-04-07 07:30:20 +0000381 for (ALL_LIST_ELEMENTS_RO (or->paths, node, path))
paul72357f22003-06-19 02:11:23 +0000382 {
paulcf795c52003-06-19 02:13:25 +0000383 if (path->nexthop.s_addr != INADDR_ANY)
384 {
385 stream_putc (s, ZEBRA_NEXTHOP_IPV4);
386 stream_put_in_addr (s, &path->nexthop);
387 }
388 else
389 {
390 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
391 if (path->oi)
392 stream_putl (s, path->oi->ifp->ifindex);
393 else
394 stream_putl (s, 0);
395 }
paul72357f22003-06-19 02:11:23 +0000396
397 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
398 {
ajs9b0e25c2004-12-08 19:06:51 +0000399 zlog_debug ("Zebra: Route add %s/%d nexthop %s",
paulcf795c52003-06-19 02:13:25 +0000400 inet_ntoa (p->prefix),
401 p->prefixlen, inet_ntoa (path->nexthop));
402 }
403 }
paul718e3742002-12-13 20:15:29 +0000404
405 if (CHECK_FLAG (message, ZAPI_MESSAGE_DISTANCE))
paulcf795c52003-06-19 02:13:25 +0000406 stream_putc (s, distance);
paul718e3742002-12-13 20:15:29 +0000407 if (CHECK_FLAG (message, ZAPI_MESSAGE_METRIC))
paulcf795c52003-06-19 02:13:25 +0000408 {
409 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL)
410 stream_putl (s, or->cost + or->u.ext.type2_cost);
411 else if (or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
412 stream_putl (s, or->u.ext.type2_cost);
413 else
414 stream_putl (s, or->cost);
415 }
paul718e3742002-12-13 20:15:29 +0000416
417 stream_putw_at (s, 0, stream_get_endp (s));
418
ajs634f9ea2005-04-11 15:51:40 +0000419 zclient_send_message(zclient);
paul718e3742002-12-13 20:15:29 +0000420 }
421}
422
423void
424ospf_zebra_delete (struct prefix_ipv4 *p, struct ospf_route *or)
425{
426 struct zapi_ipv4 api;
paul72357f22003-06-19 02:11:23 +0000427 struct ospf_path *path;
428 struct in_addr *nexthop;
paul1eb8ef22005-04-07 07:30:20 +0000429 struct listnode *node, *nnode;
paul718e3742002-12-13 20:15:29 +0000430
431 if (zclient->redist[ZEBRA_ROUTE_OSPF])
432 {
433 api.type = ZEBRA_ROUTE_OSPF;
434 api.flags = 0;
435 api.message = 0;
paul72357f22003-06-19 02:11:23 +0000436 api.ifindex_num = 0;
437 api.nexthop_num = 0;
paul718e3742002-12-13 20:15:29 +0000438
paul1eb8ef22005-04-07 07:30:20 +0000439 for (ALL_LIST_ELEMENTS (or->paths, node, nnode, path))
paulcf795c52003-06-19 02:13:25 +0000440 {
paulcf795c52003-06-19 02:13:25 +0000441 if (path->nexthop.s_addr != INADDR_ANY)
442 {
pauld8e1d6b2003-08-10 04:04:41 +0000443 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
paulcf795c52003-06-19 02:13:25 +0000444 api.nexthop_num = 1;
445 nexthop = &path->nexthop;
446 api.nexthop = &nexthop;
447 }
hasso2db3d052004-02-11 21:52:13 +0000448 else if (ospf_if_exists(path->oi) && (path->oi->ifp))
paulbb8ff1e2003-08-12 06:00:30 +0000449 {
450 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
451 api.ifindex_num = 1;
452 api.ifindex = &path->oi->ifp->ifindex;
453 }
454 else if ( IS_DEBUG_OSPF(zebra,ZEBRA_REDISTRIBUTE) )
455 {
ajs9b0e25c2004-12-08 19:06:51 +0000456 zlog_debug("Zebra: no ifp %s %d",
paulbb8ff1e2003-08-12 06:00:30 +0000457 inet_ntoa(p->prefix),
458 p->prefixlen);
459 }
paul72357f22003-06-19 02:11:23 +0000460
paul0a589352004-05-08 11:48:26 +0000461 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
paul72357f22003-06-19 02:11:23 +0000462
463 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.nexthop_num)
paulcf795c52003-06-19 02:13:25 +0000464 {
ajs9b0e25c2004-12-08 19:06:51 +0000465 zlog_debug ("Zebra: Route delete %s/%d nexthop %s",
paulcf795c52003-06-19 02:13:25 +0000466 inet_ntoa (p->prefix),
467 p->prefixlen, inet_ntoa (**api.nexthop));
468 }
paulbb8ff1e2003-08-12 06:00:30 +0000469 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.ifindex_num)
470 {
ajs9b0e25c2004-12-08 19:06:51 +0000471 zlog_debug ("Zebra: Route delete %s/%d ifindex %d",
paulbb8ff1e2003-08-12 06:00:30 +0000472 inet_ntoa (p->prefix),
473 p->prefixlen, *api.ifindex);
474 }
paulcf795c52003-06-19 02:13:25 +0000475 }
paul718e3742002-12-13 20:15:29 +0000476 }
477}
478
479void
480ospf_zebra_add_discard (struct prefix_ipv4 *p)
481{
482 struct zapi_ipv4 api;
483
484 if (zclient->redist[ZEBRA_ROUTE_OSPF])
485 {
486 api.type = ZEBRA_ROUTE_OSPF;
487 api.flags = ZEBRA_FLAG_BLACKHOLE;
488 api.message = 0;
489 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
490 api.nexthop_num = 0;
491 api.ifindex_num = 0;
492
paul0a589352004-05-08 11:48:26 +0000493 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_ADD, zclient, p, &api);
Andrew J. Schorr7f643eb2006-11-30 16:17:02 +0000494
495 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
496 zlog_debug ("Zebra: Route add discard %s/%d",
497 inet_ntoa (p->prefix), p->prefixlen);
paul718e3742002-12-13 20:15:29 +0000498 }
499}
500
501void
502ospf_zebra_delete_discard (struct prefix_ipv4 *p)
503{
504 struct zapi_ipv4 api;
505
506 if (zclient->redist[ZEBRA_ROUTE_OSPF])
507 {
508 api.type = ZEBRA_ROUTE_OSPF;
509 api.flags = ZEBRA_FLAG_BLACKHOLE;
510 api.message = 0;
511 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
512 api.nexthop_num = 0;
513 api.ifindex_num = 0;
514
paul0a589352004-05-08 11:48:26 +0000515 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
paul72357f22003-06-19 02:11:23 +0000516
517 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000518 zlog_debug ("Zebra: Route delete discard %s/%d",
paulcf795c52003-06-19 02:13:25 +0000519 inet_ntoa (p->prefix), p->prefixlen);
paul72357f22003-06-19 02:11:23 +0000520
paul718e3742002-12-13 20:15:29 +0000521 }
522}
523
524int
525ospf_is_type_redistributed (int type)
526{
527 return (DEFAULT_ROUTE_TYPE (type)) ?
528 zclient->default_information : zclient->redist[type];
529}
530
531int
paul020709f2003-04-04 02:44:16 +0000532ospf_redistribute_set (struct ospf *ospf, int type, int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000533{
534 int force = 0;
paulcf795c52003-06-19 02:13:25 +0000535
paul718e3742002-12-13 20:15:29 +0000536 if (ospf_is_type_redistributed (type))
537 {
paul68980082003-03-25 05:07:42 +0000538 if (mtype != ospf->dmetric[type].type)
paulcf795c52003-06-19 02:13:25 +0000539 {
540 ospf->dmetric[type].type = mtype;
541 force = LSA_REFRESH_FORCE;
542 }
paul68980082003-03-25 05:07:42 +0000543 if (mvalue != ospf->dmetric[type].value)
paulcf795c52003-06-19 02:13:25 +0000544 {
545 ospf->dmetric[type].value = mvalue;
546 force = LSA_REFRESH_FORCE;
547 }
548
paul68980082003-03-25 05:07:42 +0000549 ospf_external_lsa_refresh_type (ospf, type, force);
paulcf795c52003-06-19 02:13:25 +0000550
paul718e3742002-12-13 20:15:29 +0000551 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000552 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000553 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000554 metric_type (ospf, type), metric_value (ospf, type));
555
paul718e3742002-12-13 20:15:29 +0000556 return CMD_SUCCESS;
557 }
558
paul68980082003-03-25 05:07:42 +0000559 ospf->dmetric[type].type = mtype;
560 ospf->dmetric[type].value = mvalue;
paul718e3742002-12-13 20:15:29 +0000561
paul0a589352004-05-08 11:48:26 +0000562 zclient_redistribute (ZEBRA_REDISTRIBUTE_ADD, zclient, type);
paul718e3742002-12-13 20:15:29 +0000563
564 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000565 zlog_debug ("Redistribute[%s]: Start Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000566 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000567 metric_type (ospf, type), metric_value (ospf, type));
568
paul68980082003-03-25 05:07:42 +0000569 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000570
571 return CMD_SUCCESS;
572}
573
574int
paul020709f2003-04-04 02:44:16 +0000575ospf_redistribute_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000576{
577 if (type == zclient->redist_default)
578 return CMD_SUCCESS;
579
paulcf795c52003-06-19 02:13:25 +0000580 if (!ospf_is_type_redistributed (type))
paul718e3742002-12-13 20:15:29 +0000581 return CMD_SUCCESS;
582
paul0a589352004-05-08 11:48:26 +0000583 zclient_redistribute (ZEBRA_REDISTRIBUTE_DELETE, zclient, type);
paulcf795c52003-06-19 02:13:25 +0000584
paul718e3742002-12-13 20:15:29 +0000585 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000586 zlog_debug ("Redistribute[%s]: Stop",
ajsf52d13c2005-10-01 17:38:06 +0000587 ospf_redist_string(type));
paul718e3742002-12-13 20:15:29 +0000588
paul68980082003-03-25 05:07:42 +0000589 ospf->dmetric[type].type = -1;
590 ospf->dmetric[type].value = -1;
paul718e3742002-12-13 20:15:29 +0000591
592 /* Remove the routes from OSPF table. */
Paul Jakma6db3a6f2006-05-12 23:02:46 +0000593 ospf_redistribute_withdraw (ospf, type);
paul718e3742002-12-13 20:15:29 +0000594
paul68980082003-03-25 05:07:42 +0000595 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000596
597 return CMD_SUCCESS;
598}
599
600int
paul020709f2003-04-04 02:44:16 +0000601ospf_redistribute_default_set (struct ospf *ospf, int originate,
paulcf795c52003-06-19 02:13:25 +0000602 int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000603{
Andrew J. Schorr8fb8a502006-10-24 19:04:26 +0000604 ospf->default_originate = originate;
605 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
606 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
paul020709f2003-04-04 02:44:16 +0000607
paul718e3742002-12-13 20:15:29 +0000608 if (ospf_is_type_redistributed (DEFAULT_ROUTE))
609 {
Andrew J. Schorr8fb8a502006-10-24 19:04:26 +0000610 /* if ospf->default_originate changes value, is calling
611 ospf_external_lsa_refresh_default sufficient to implement
612 the change? */
paul68980082003-03-25 05:07:42 +0000613 ospf_external_lsa_refresh_default (ospf);
paulcf795c52003-06-19 02:13:25 +0000614
paul718e3742002-12-13 20:15:29 +0000615 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000616 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000617 ospf_redist_string(DEFAULT_ROUTE),
paulcf795c52003-06-19 02:13:25 +0000618 metric_type (ospf, DEFAULT_ROUTE),
619 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000620 return CMD_SUCCESS;
621 }
622
paul0a589352004-05-08 11:48:26 +0000623 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_ADD, zclient);
paulcf795c52003-06-19 02:13:25 +0000624
paul718e3742002-12-13 20:15:29 +0000625 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000626 zlog_debug ("Redistribute[DEFAULT]: Start Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000627 metric_type (ospf, DEFAULT_ROUTE),
628 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000629
paul68980082003-03-25 05:07:42 +0000630 if (ospf->router_id.s_addr == 0)
631 ospf->external_origin |= (1 << DEFAULT_ROUTE);
paul718e3742002-12-13 20:15:29 +0000632 else
Paul Jakma4021b602006-05-12 22:55:41 +0000633 thread_add_timer (master, ospf_default_originate_timer, ospf, 1);
paul718e3742002-12-13 20:15:29 +0000634
paul68980082003-03-25 05:07:42 +0000635 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000636
637 return CMD_SUCCESS;
638}
639
640int
paul020709f2003-04-04 02:44:16 +0000641ospf_redistribute_default_unset (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +0000642{
643 if (!ospf_is_type_redistributed (DEFAULT_ROUTE))
644 return CMD_SUCCESS;
645
paul68980082003-03-25 05:07:42 +0000646 ospf->default_originate = DEFAULT_ORIGINATE_NONE;
647 ospf->dmetric[DEFAULT_ROUTE].type = -1;
648 ospf->dmetric[DEFAULT_ROUTE].value = -1;
paul718e3742002-12-13 20:15:29 +0000649
paul0a589352004-05-08 11:48:26 +0000650 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_DELETE, zclient);
paul718e3742002-12-13 20:15:29 +0000651
652 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000653 zlog_debug ("Redistribute[DEFAULT]: Stop");
paulcf795c52003-06-19 02:13:25 +0000654
paul68980082003-03-25 05:07:42 +0000655 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000656
657 return CMD_SUCCESS;
658}
659
paul4dadc292005-05-06 21:37:42 +0000660static int
paul020709f2003-04-04 02:44:16 +0000661ospf_external_lsa_originate_check (struct ospf *ospf,
paulcf795c52003-06-19 02:13:25 +0000662 struct external_info *ei)
paul718e3742002-12-13 20:15:29 +0000663{
664 /* If prefix is multicast, then do not originate LSA. */
665 if (IN_MULTICAST (htonl (ei->p.prefix.s_addr)))
666 {
667 zlog_info ("LSA[Type5:%s]: Not originate AS-external-LSA, "
paulcf795c52003-06-19 02:13:25 +0000668 "Prefix belongs multicast", inet_ntoa (ei->p.prefix));
paul718e3742002-12-13 20:15:29 +0000669 return 0;
670 }
671
672 /* Take care of default-originate. */
673 if (is_prefix_default (&ei->p))
paul68980082003-03-25 05:07:42 +0000674 if (ospf->default_originate == DEFAULT_ORIGINATE_NONE)
paul718e3742002-12-13 20:15:29 +0000675 {
paulcf795c52003-06-19 02:13:25 +0000676 zlog_info ("LSA[Type5:0.0.0.0]: Not originate AS-exntenal-LSA "
677 "for default");
678 return 0;
paul718e3742002-12-13 20:15:29 +0000679 }
680
681 return 1;
682}
683
684/* If connected prefix is OSPF enable interface, then do not announce. */
685int
paulcf795c52003-06-19 02:13:25 +0000686ospf_distribute_check_connected (struct ospf *ospf, struct external_info *ei)
paul718e3742002-12-13 20:15:29 +0000687{
688 struct route_node *rn;
689
paul68980082003-03-25 05:07:42 +0000690 for (rn = route_top (ospf->networks); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +0000691 if (rn->info != NULL)
paulcf795c52003-06-19 02:13:25 +0000692 if (prefix_match (&rn->p, (struct prefix *) &ei->p))
693 {
694 route_unlock_node (rn);
695 return 0;
696 }
paul718e3742002-12-13 20:15:29 +0000697
698 return 1;
699}
700
701/* return 1 if external LSA must be originated, 0 otherwise */
702int
paul68980082003-03-25 05:07:42 +0000703ospf_redistribute_check (struct ospf *ospf,
paulcf795c52003-06-19 02:13:25 +0000704 struct external_info *ei, int *changed)
paul718e3742002-12-13 20:15:29 +0000705{
706 struct route_map_set_values save_values;
707 struct prefix_ipv4 *p = &ei->p;
708 u_char type = is_prefix_default (&ei->p) ? DEFAULT_ROUTE : ei->type;
paulcf795c52003-06-19 02:13:25 +0000709
paul718e3742002-12-13 20:15:29 +0000710 if (changed)
711 *changed = 0;
712
paul020709f2003-04-04 02:44:16 +0000713 if (!ospf_external_lsa_originate_check (ospf, ei))
paul718e3742002-12-13 20:15:29 +0000714 return 0;
715
716 /* Take care connected route. */
paul68980082003-03-25 05:07:42 +0000717 if (type == ZEBRA_ROUTE_CONNECT &&
718 !ospf_distribute_check_connected (ospf, ei))
paul718e3742002-12-13 20:15:29 +0000719 return 0;
720
paul020709f2003-04-04 02:44:16 +0000721 if (!DEFAULT_ROUTE_TYPE (type) && DISTRIBUTE_NAME (ospf, type))
paul718e3742002-12-13 20:15:29 +0000722 /* distirbute-list exists, but access-list may not? */
paul020709f2003-04-04 02:44:16 +0000723 if (DISTRIBUTE_LIST (ospf, type))
724 if (access_list_apply (DISTRIBUTE_LIST (ospf, type), p) == FILTER_DENY)
paulcf795c52003-06-19 02:13:25 +0000725 {
726 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000727 zlog_debug ("Redistribute[%s]: %s/%d filtered by ditribute-list.",
ajsf52d13c2005-10-01 17:38:06 +0000728 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000729 inet_ntoa (p->prefix), p->prefixlen);
730 return 0;
731 }
paul718e3742002-12-13 20:15:29 +0000732
733 save_values = ei->route_map_set;
734 ospf_reset_route_map_set_values (&ei->route_map_set);
paulcf795c52003-06-19 02:13:25 +0000735
paul718e3742002-12-13 20:15:29 +0000736 /* apply route-map if needed */
paul020709f2003-04-04 02:44:16 +0000737 if (ROUTEMAP_NAME (ospf, type))
paul718e3742002-12-13 20:15:29 +0000738 {
739 int ret;
740
paulcf795c52003-06-19 02:13:25 +0000741 ret = route_map_apply (ROUTEMAP (ospf, type), (struct prefix *) p,
742 RMAP_OSPF, ei);
paul718e3742002-12-13 20:15:29 +0000743
744 if (ret == RMAP_DENYMATCH)
paulcf795c52003-06-19 02:13:25 +0000745 {
746 ei->route_map_set = save_values;
747 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000748 zlog_debug ("Redistribute[%s]: %s/%d filtered by route-map.",
ajsf52d13c2005-10-01 17:38:06 +0000749 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000750 inet_ntoa (p->prefix), p->prefixlen);
751 return 0;
752 }
753
paul718e3742002-12-13 20:15:29 +0000754 /* check if 'route-map set' changed something */
755 if (changed)
paulcf795c52003-06-19 02:13:25 +0000756 *changed = !ospf_route_map_set_compare (&ei->route_map_set,
757 &save_values);
paul718e3742002-12-13 20:15:29 +0000758 }
759
760 return 1;
761}
762
763/* OSPF route-map set for redistribution */
764void
paul6c835672004-10-11 11:00:30 +0000765ospf_routemap_set (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000766{
paul020709f2003-04-04 02:44:16 +0000767 if (ROUTEMAP_NAME (ospf, type))
768 free (ROUTEMAP_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000769
paul020709f2003-04-04 02:44:16 +0000770 ROUTEMAP_NAME (ospf, type) = strdup (name);
771 ROUTEMAP (ospf, type) = route_map_lookup_by_name (name);
paul718e3742002-12-13 20:15:29 +0000772}
773
774void
paul020709f2003-04-04 02:44:16 +0000775ospf_routemap_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000776{
paul020709f2003-04-04 02:44:16 +0000777 if (ROUTEMAP_NAME (ospf, type))
778 free (ROUTEMAP_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000779
paul020709f2003-04-04 02:44:16 +0000780 ROUTEMAP_NAME (ospf, type) = NULL;
781 ROUTEMAP (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000782}
783
784/* Zebra route add and delete treatment. */
paul4dadc292005-05-06 21:37:42 +0000785static int
paul718e3742002-12-13 20:15:29 +0000786ospf_zebra_read_ipv4 (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000787 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000788{
789 struct stream *s;
790 struct zapi_ipv4 api;
791 unsigned long ifindex;
792 struct in_addr nexthop;
793 struct prefix_ipv4 p;
794 struct external_info *ei;
paul020709f2003-04-04 02:44:16 +0000795 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000796
797 s = zclient->ibuf;
798 ifindex = 0;
799 nexthop.s_addr = 0;
800
801 /* Type, flags, message. */
802 api.type = stream_getc (s);
803 api.flags = stream_getc (s);
804 api.message = stream_getc (s);
805
806 /* IPv4 prefix. */
807 memset (&p, 0, sizeof (struct prefix_ipv4));
808 p.family = AF_INET;
809 p.prefixlen = stream_getc (s);
810 stream_get (&p.prefix, s, PSIZE (p.prefixlen));
811
hasso8585d4e2004-04-20 17:25:12 +0000812 if (IPV4_NET127(ntohl(p.prefix.s_addr)))
813 return 0;
814
paul718e3742002-12-13 20:15:29 +0000815 /* Nexthop, ifindex, distance, metric. */
816 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP))
817 {
818 api.nexthop_num = stream_getc (s);
819 nexthop.s_addr = stream_get_ipv4 (s);
820 }
821 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_IFINDEX))
822 {
823 api.ifindex_num = stream_getc (s);
gdt6a8da852004-02-13 17:40:51 +0000824 /* XXX assert(api.ifindex_num == 1); */
paul718e3742002-12-13 20:15:29 +0000825 ifindex = stream_getl (s);
826 }
827 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_DISTANCE))
828 api.distance = stream_getc (s);
829 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_METRIC))
830 api.metric = stream_getl (s);
831
paul020709f2003-04-04 02:44:16 +0000832 ospf = ospf_lookup ();
833 if (ospf == NULL)
834 return 0;
835
paul718e3742002-12-13 20:15:29 +0000836 if (command == ZEBRA_IPV4_ROUTE_ADD)
837 {
paul7021c422003-07-15 12:52:22 +0000838 /* XXX|HACK|TODO|FIXME:
hassoa0a39762004-04-23 08:51:10 +0000839 * Maybe we should ignore reject/blackhole routes? Testing shows that
840 * there is no problems though and this is only way to "summarize"
841 * routes in ASBR at the moment. Maybe we need just a better generalised
842 * solution for these types?
843 *
844 * if ( CHECK_FLAG (api.flags, ZEBRA_FLAG_BLACKHOLE)
845 * || CHECK_FLAG (api.flags, ZEBRA_FLAG_REJECT))
846 * return 0;
paul7021c422003-07-15 12:52:22 +0000847 */
paul7021c422003-07-15 12:52:22 +0000848
paul718e3742002-12-13 20:15:29 +0000849 ei = ospf_external_info_add (api.type, p, ifindex, nexthop);
850
paul68980082003-03-25 05:07:42 +0000851 if (ospf->router_id.s_addr == 0)
paulcf795c52003-06-19 02:13:25 +0000852 /* Set flags to generate AS-external-LSA originate event
853 for each redistributed protocols later. */
854 ospf->external_origin |= (1 << api.type);
paul718e3742002-12-13 20:15:29 +0000855 else
paulcf795c52003-06-19 02:13:25 +0000856 {
857 if (ei)
858 {
859 if (is_prefix_default (&p))
860 ospf_external_lsa_refresh_default (ospf);
861 else
862 {
863 struct ospf_lsa *current;
paul718e3742002-12-13 20:15:29 +0000864
paulcf795c52003-06-19 02:13:25 +0000865 current = ospf_external_info_find_lsa (ospf, &ei->p);
866 if (!current)
867 ospf_external_lsa_originate (ospf, ei);
868 else if (IS_LSA_MAXAGE (current))
869 ospf_external_lsa_refresh (ospf, current,
870 ei, LSA_REFRESH_FORCE);
871 else
872 zlog_warn ("ospf_zebra_read_ipv4() : %s already exists",
873 inet_ntoa (p.prefix));
874 }
875 }
876 }
paul718e3742002-12-13 20:15:29 +0000877 }
paulcf795c52003-06-19 02:13:25 +0000878 else /* if (command == ZEBRA_IPV4_ROUTE_DELETE) */
paul718e3742002-12-13 20:15:29 +0000879 {
880 ospf_external_info_delete (api.type, p);
paul7021c422003-07-15 12:52:22 +0000881 if (is_prefix_default (&p))
paulcf795c52003-06-19 02:13:25 +0000882 ospf_external_lsa_refresh_default (ospf);
paul7021c422003-07-15 12:52:22 +0000883 else
ajs5339cfd2005-09-19 13:28:05 +0000884 ospf_external_lsa_flush (ospf, api.type, &p, ifindex /*, nexthop */);
paul718e3742002-12-13 20:15:29 +0000885 }
886
887 return 0;
888}
paul718e3742002-12-13 20:15:29 +0000889
paulcf795c52003-06-19 02:13:25 +0000890
paul718e3742002-12-13 20:15:29 +0000891int
paul6c835672004-10-11 11:00:30 +0000892ospf_distribute_list_out_set (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000893{
894 /* Lookup access-list for distribute-list. */
paul020709f2003-04-04 02:44:16 +0000895 DISTRIBUTE_LIST (ospf, type) = access_list_lookup (AFI_IP, name);
paul718e3742002-12-13 20:15:29 +0000896
897 /* Clear previous distribute-name. */
paul020709f2003-04-04 02:44:16 +0000898 if (DISTRIBUTE_NAME (ospf, type))
899 free (DISTRIBUTE_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000900
901 /* Set distribute-name. */
paul020709f2003-04-04 02:44:16 +0000902 DISTRIBUTE_NAME (ospf, type) = strdup (name);
paul718e3742002-12-13 20:15:29 +0000903
904 /* If access-list have been set, schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000905 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000906 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000907
908 return CMD_SUCCESS;
909}
910
911int
paul6c835672004-10-11 11:00:30 +0000912ospf_distribute_list_out_unset (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000913{
914 /* Schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000915 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000916 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000917
918 /* Unset distribute-list. */
paul020709f2003-04-04 02:44:16 +0000919 DISTRIBUTE_LIST (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000920
921 /* Clear distribute-name. */
paul020709f2003-04-04 02:44:16 +0000922 if (DISTRIBUTE_NAME (ospf, type))
923 free (DISTRIBUTE_NAME (ospf, type));
paulcf795c52003-06-19 02:13:25 +0000924
paul020709f2003-04-04 02:44:16 +0000925 DISTRIBUTE_NAME (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000926
927 return CMD_SUCCESS;
928}
929
930/* distribute-list update timer. */
paul4dadc292005-05-06 21:37:42 +0000931static int
paul718e3742002-12-13 20:15:29 +0000932ospf_distribute_list_update_timer (struct thread *thread)
933{
934 struct route_node *rn;
935 struct external_info *ei;
936 struct route_table *rt;
937 struct ospf_lsa *lsa;
paul4dadc292005-05-06 21:37:42 +0000938 intptr_t type;
paul020709f2003-04-04 02:44:16 +0000939 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000940
paul4dadc292005-05-06 21:37:42 +0000941 type = (intptr_t)THREAD_ARG (thread);
hasso01018ce2005-08-05 07:40:15 +0000942 assert (type <= ZEBRA_ROUTE_MAX);
paul64511f32004-10-31 18:01:13 +0000943
paul718e3742002-12-13 20:15:29 +0000944 rt = EXTERNAL_INFO (type);
945
paul020709f2003-04-04 02:44:16 +0000946 ospf = ospf_lookup ();
947 if (ospf == NULL)
948 return 0;
949
paul68980082003-03-25 05:07:42 +0000950 ospf->t_distribute_update = NULL;
paul718e3742002-12-13 20:15:29 +0000951
952 zlog_info ("Zebra[Redistribute]: distribute-list update timer fired!");
953
954 /* foreach all external info. */
955 if (rt)
956 for (rn = route_top (rt); rn; rn = route_next (rn))
957 if ((ei = rn->info) != NULL)
paulcf795c52003-06-19 02:13:25 +0000958 {
959 if (is_prefix_default (&ei->p))
960 ospf_external_lsa_refresh_default (ospf);
961 else if ((lsa = ospf_external_info_find_lsa (ospf, &ei->p)))
962 ospf_external_lsa_refresh (ospf, lsa, ei, LSA_REFRESH_IF_CHANGED);
963 else
964 ospf_external_lsa_originate (ospf, ei);
965 }
paul718e3742002-12-13 20:15:29 +0000966 return 0;
967}
968
969#define OSPF_DISTRIBUTE_UPDATE_DELAY 5
970
971/* Update distribute-list and set timer to apply access-list. */
972void
paul68980082003-03-25 05:07:42 +0000973ospf_distribute_list_update (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000974{
975 struct route_table *rt;
paulcf795c52003-06-19 02:13:25 +0000976
paul718e3742002-12-13 20:15:29 +0000977 /* External info does not exist. */
978 if (!(rt = EXTERNAL_INFO (type)))
979 return;
980
981 /* If exists previously invoked thread, then cancel it. */
paul68980082003-03-25 05:07:42 +0000982 if (ospf->t_distribute_update)
983 OSPF_TIMER_OFF (ospf->t_distribute_update);
paul718e3742002-12-13 20:15:29 +0000984
985 /* Set timer. */
paul68980082003-03-25 05:07:42 +0000986 ospf->t_distribute_update =
paul718e3742002-12-13 20:15:29 +0000987 thread_add_timer (master, ospf_distribute_list_update_timer,
paulcf795c52003-06-19 02:13:25 +0000988 (void *) type, OSPF_DISTRIBUTE_UPDATE_DELAY);
paul718e3742002-12-13 20:15:29 +0000989}
990
991/* If access-list is updated, apply some check. */
paul4dadc292005-05-06 21:37:42 +0000992static void
paul718e3742002-12-13 20:15:29 +0000993ospf_filter_update (struct access_list *access)
994{
paul020709f2003-04-04 02:44:16 +0000995 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000996 int type;
997 int abr_inv = 0;
998 struct ospf_area *area;
hasso52dc7ee2004-09-23 19:18:23 +0000999 struct listnode *node;
paul718e3742002-12-13 20:15:29 +00001000
1001 /* If OSPF instatnce does not exist, return right now. */
paul020709f2003-04-04 02:44:16 +00001002 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +00001003 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +00001004 return;
1005
paul718e3742002-12-13 20:15:29 +00001006 /* Update distribute-list, and apply filter. */
hasso01018ce2005-08-05 07:40:15 +00001007 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
paul718e3742002-12-13 20:15:29 +00001008 {
paul020709f2003-04-04 02:44:16 +00001009 if (ROUTEMAP (ospf, type) != NULL)
paulcf795c52003-06-19 02:13:25 +00001010 {
1011 /* if route-map is not NULL it may be using this access list */
1012 ospf_distribute_list_update (ospf, type);
1013 continue;
1014 }
1015
hasso01018ce2005-08-05 07:40:15 +00001016 /* There is place for route-map for default-information (ZEBRA_ROUTE_MAX),
1017 * but no distribute list. */
1018 if (type == ZEBRA_ROUTE_MAX)
1019 break;
paul718e3742002-12-13 20:15:29 +00001020
paul020709f2003-04-04 02:44:16 +00001021 if (DISTRIBUTE_NAME (ospf, type))
paulcf795c52003-06-19 02:13:25 +00001022 {
1023 /* Keep old access-list for distribute-list. */
1024 struct access_list *old = DISTRIBUTE_LIST (ospf, type);
1025
1026 /* Update access-list for distribute-list. */
1027 DISTRIBUTE_LIST (ospf, type) =
1028 access_list_lookup (AFI_IP, DISTRIBUTE_NAME (ospf, type));
1029
1030 /* No update for this distribute type. */
1031 if (old == NULL && DISTRIBUTE_LIST (ospf, type) == NULL)
1032 continue;
1033
1034 /* Schedule distribute-list update timer. */
1035 if (DISTRIBUTE_LIST (ospf, type) == NULL ||
1036 strcmp (DISTRIBUTE_NAME (ospf, type), access->name) == 0)
1037 ospf_distribute_list_update (ospf, type);
1038 }
paul718e3742002-12-13 20:15:29 +00001039 }
1040
1041 /* Update Area access-list. */
paul1eb8ef22005-04-07 07:30:20 +00001042 for (ALL_LIST_ELEMENTS_RO (ospf->areas, node, area))
1043 {
1044 if (EXPORT_NAME (area))
1045 {
1046 EXPORT_LIST (area) = NULL;
1047 abr_inv++;
1048 }
paul718e3742002-12-13 20:15:29 +00001049
paul1eb8ef22005-04-07 07:30:20 +00001050 if (IMPORT_NAME (area))
1051 {
1052 IMPORT_LIST (area) = NULL;
1053 abr_inv++;
1054 }
1055 }
paul718e3742002-12-13 20:15:29 +00001056
1057 /* Schedule ABR tasks -- this will be changed -- takada. */
paul020709f2003-04-04 02:44:16 +00001058 if (IS_OSPF_ABR (ospf) && abr_inv)
paul68980082003-03-25 05:07:42 +00001059 ospf_schedule_abr_task (ospf);
paul718e3742002-12-13 20:15:29 +00001060}
hassodd669bb2004-05-10 07:43:59 +00001061
1062/* If prefix-list is updated, do some updates. */
1063void
1064ospf_prefix_list_update (struct prefix_list *plist)
1065{
1066 struct ospf *ospf;
1067 int type;
1068 int abr_inv = 0;
1069 struct ospf_area *area;
hasso52dc7ee2004-09-23 19:18:23 +00001070 struct listnode *node;
hassodd669bb2004-05-10 07:43:59 +00001071
1072 /* If OSPF instatnce does not exist, return right now. */
1073 ospf = ospf_lookup ();
1074 if (ospf == NULL)
1075 return;
1076
1077 /* Update all route-maps which are used as redistribution filters.
1078 * They might use prefix-list.
1079 */
hasso01018ce2005-08-05 07:40:15 +00001080 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
hassodd669bb2004-05-10 07:43:59 +00001081 {
1082 if (ROUTEMAP (ospf, type) != NULL)
1083 {
1084 /* If route-map is not NULL it may be using this prefix list */
1085 ospf_distribute_list_update (ospf, type);
1086 continue;
1087 }
1088 }
1089
1090 /* Update area filter-lists. */
paul1eb8ef22005-04-07 07:30:20 +00001091 for (ALL_LIST_ELEMENTS_RO (ospf->areas, node, area))
1092 {
1093 /* Update filter-list in. */
1094 if (PREFIX_NAME_IN (area))
1095 if (strcmp (PREFIX_NAME_IN (area), plist->name) == 0)
1096 {
1097 PREFIX_LIST_IN (area) =
1098 prefix_list_lookup (AFI_IP, PREFIX_NAME_IN (area));
1099 abr_inv++;
1100 }
hassodd669bb2004-05-10 07:43:59 +00001101
paul1eb8ef22005-04-07 07:30:20 +00001102 /* Update filter-list out. */
1103 if (PREFIX_NAME_OUT (area))
1104 if (strcmp (PREFIX_NAME_OUT (area), plist->name) == 0)
1105 {
1106 PREFIX_LIST_IN (area) =
1107 prefix_list_lookup (AFI_IP, PREFIX_NAME_OUT (area));
1108 abr_inv++;
1109 }
1110 }
hassodd669bb2004-05-10 07:43:59 +00001111
1112 /* Schedule ABR task. */
1113 if (IS_OSPF_ABR (ospf) && abr_inv)
1114 ospf_schedule_abr_task (ospf);
1115}
paulcf795c52003-06-19 02:13:25 +00001116
paul4dadc292005-05-06 21:37:42 +00001117static struct ospf_distance *
1118ospf_distance_new (void)
paul718e3742002-12-13 20:15:29 +00001119{
1120 struct ospf_distance *new;
1121 new = XMALLOC (MTYPE_OSPF_DISTANCE, sizeof (struct ospf_distance));
1122 memset (new, 0, sizeof (struct ospf_distance));
1123 return new;
1124}
1125
paul4dadc292005-05-06 21:37:42 +00001126static void
paul718e3742002-12-13 20:15:29 +00001127ospf_distance_free (struct ospf_distance *odistance)
1128{
1129 XFREE (MTYPE_OSPF_DISTANCE, odistance);
1130}
1131
1132int
paul6c835672004-10-11 11:00:30 +00001133ospf_distance_set (struct vty *vty, struct ospf *ospf,
1134 const char *distance_str,
1135 const char *ip_str,
1136 const char *access_list_str)
paul718e3742002-12-13 20:15:29 +00001137{
1138 int ret;
1139 struct prefix_ipv4 p;
1140 u_char distance;
1141 struct route_node *rn;
1142 struct ospf_distance *odistance;
1143
1144 ret = str2prefix_ipv4 (ip_str, &p);
1145 if (ret == 0)
1146 {
1147 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1148 return CMD_WARNING;
1149 }
1150
1151 distance = atoi (distance_str);
1152
1153 /* Get OSPF distance node. */
paul68980082003-03-25 05:07:42 +00001154 rn = route_node_get (ospf->distance_table, (struct prefix *) &p);
paul718e3742002-12-13 20:15:29 +00001155 if (rn->info)
1156 {
1157 odistance = rn->info;
1158 route_unlock_node (rn);
1159 }
1160 else
1161 {
1162 odistance = ospf_distance_new ();
1163 rn->info = odistance;
1164 }
1165
1166 /* Set distance value. */
1167 odistance->distance = distance;
1168
1169 /* Reset access-list configuration. */
1170 if (odistance->access_list)
1171 {
1172 free (odistance->access_list);
1173 odistance->access_list = NULL;
1174 }
1175 if (access_list_str)
1176 odistance->access_list = strdup (access_list_str);
1177
1178 return CMD_SUCCESS;
1179}
1180
1181int
paul6c835672004-10-11 11:00:30 +00001182ospf_distance_unset (struct vty *vty, struct ospf *ospf,
1183 const char *distance_str,
1184 const char *ip_str, char
1185 const *access_list_str)
paul718e3742002-12-13 20:15:29 +00001186{
1187 int ret;
1188 struct prefix_ipv4 p;
1189 u_char distance;
1190 struct route_node *rn;
1191 struct ospf_distance *odistance;
1192
1193 ret = str2prefix_ipv4 (ip_str, &p);
1194 if (ret == 0)
1195 {
1196 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1197 return CMD_WARNING;
1198 }
1199
1200 distance = atoi (distance_str);
1201
paulcf795c52003-06-19 02:13:25 +00001202 rn = route_node_lookup (ospf->distance_table, (struct prefix *) &p);
1203 if (!rn)
paul718e3742002-12-13 20:15:29 +00001204 {
1205 vty_out (vty, "Can't find specified prefix%s", VTY_NEWLINE);
1206 return CMD_WARNING;
1207 }
1208
1209 odistance = rn->info;
1210
1211 if (odistance->access_list)
1212 free (odistance->access_list);
1213 ospf_distance_free (odistance);
1214
1215 rn->info = NULL;
1216 route_unlock_node (rn);
1217 route_unlock_node (rn);
1218
1219 return CMD_SUCCESS;
1220}
1221
1222void
paul68980082003-03-25 05:07:42 +00001223ospf_distance_reset (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +00001224{
1225 struct route_node *rn;
1226 struct ospf_distance *odistance;
1227
paul68980082003-03-25 05:07:42 +00001228 for (rn = route_top (ospf->distance_table); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +00001229 if ((odistance = rn->info) != NULL)
1230 {
paulcf795c52003-06-19 02:13:25 +00001231 if (odistance->access_list)
1232 free (odistance->access_list);
1233 ospf_distance_free (odistance);
1234 rn->info = NULL;
1235 route_unlock_node (rn);
paul718e3742002-12-13 20:15:29 +00001236 }
1237}
1238
1239u_char
1240ospf_distance_apply (struct prefix_ipv4 *p, struct ospf_route *or)
1241{
paul020709f2003-04-04 02:44:16 +00001242 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +00001243
paul020709f2003-04-04 02:44:16 +00001244 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +00001245 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +00001246 return 0;
1247
paul68980082003-03-25 05:07:42 +00001248 if (ospf->distance_intra)
paul718e3742002-12-13 20:15:29 +00001249 if (or->path_type == OSPF_PATH_INTRA_AREA)
paul68980082003-03-25 05:07:42 +00001250 return ospf->distance_intra;
paul718e3742002-12-13 20:15:29 +00001251
paul68980082003-03-25 05:07:42 +00001252 if (ospf->distance_inter)
paul718e3742002-12-13 20:15:29 +00001253 if (or->path_type == OSPF_PATH_INTER_AREA)
paul68980082003-03-25 05:07:42 +00001254 return ospf->distance_inter;
paul718e3742002-12-13 20:15:29 +00001255
paul68980082003-03-25 05:07:42 +00001256 if (ospf->distance_external)
paul718e3742002-12-13 20:15:29 +00001257 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL
paulcf795c52003-06-19 02:13:25 +00001258 || or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
paul68980082003-03-25 05:07:42 +00001259 return ospf->distance_external;
paulcf795c52003-06-19 02:13:25 +00001260
paul68980082003-03-25 05:07:42 +00001261 if (ospf->distance_all)
1262 return ospf->distance_all;
paul718e3742002-12-13 20:15:29 +00001263
1264 return 0;
1265}
1266
1267void
1268ospf_zebra_init ()
1269{
1270 /* Allocate zebra structure. */
1271 zclient = zclient_new ();
1272 zclient_init (zclient, ZEBRA_ROUTE_OSPF);
hasso18a6dce2004-10-03 18:18:34 +00001273 zclient->router_id_update = ospf_router_id_update_zebra;
paul718e3742002-12-13 20:15:29 +00001274 zclient->interface_add = ospf_interface_add;
1275 zclient->interface_delete = ospf_interface_delete;
1276 zclient->interface_up = ospf_interface_state_up;
1277 zclient->interface_down = ospf_interface_state_down;
1278 zclient->interface_address_add = ospf_interface_address_add;
1279 zclient->interface_address_delete = ospf_interface_address_delete;
1280 zclient->ipv4_route_add = ospf_zebra_read_ipv4;
1281 zclient->ipv4_route_delete = ospf_zebra_read_ipv4;
1282
1283 access_list_add_hook (ospf_filter_update);
1284 access_list_delete_hook (ospf_filter_update);
hassodd669bb2004-05-10 07:43:59 +00001285 prefix_list_add_hook (ospf_prefix_list_update);
1286 prefix_list_delete_hook (ospf_prefix_list_update);
paul718e3742002-12-13 20:15:29 +00001287}