blob: 11c4d99b0ab329b60c70137c9d60c3370c44f4ca [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
68 router_id_zebra = router_id.u.prefix4;
69
70 ospf = ospf_lookup ();
paulb29800a2005-11-20 14:50:45 +000071
hasso18a6dce2004-10-03 18:18:34 +000072 if (ospf != NULL)
paulb29800a2005-11-20 14:50:45 +000073 ospf_router_id_update (ospf);
74
hasso18a6dce2004-10-03 18:18:34 +000075 return 0;
76}
paul718e3742002-12-13 20:15:29 +000077
78/* Inteface addition message from zebra. */
paul4dadc292005-05-06 21:37:42 +000079static int
paul718e3742002-12-13 20:15:29 +000080ospf_interface_add (int command, struct zclient *zclient, zebra_size_t length)
81{
82 struct interface *ifp;
paul020709f2003-04-04 02:44:16 +000083 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +000084
85 ifp = zebra_interface_add_read (zclient->ibuf);
86
87 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +000088 zlog_debug ("Zebra: interface add %s index %d flags %ld metric %d mtu %d",
paulcf795c52003-06-19 02:13:25 +000089 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +000090
paulcf795c52003-06-19 02:13:25 +000091 assert (ifp->info);
paulf2c80652002-12-13 21:44:27 +000092
paul718e3742002-12-13 20:15:29 +000093 if (!OSPF_IF_PARAM_CONFIGURED (IF_DEF_PARAMS (ifp), type))
94 {
95 SET_IF_PARAM (IF_DEF_PARAMS (ifp), type);
ajsbc18d612004-12-15 15:07:19 +000096 IF_DEF_PARAMS (ifp)->type = ospf_default_iftype(ifp);
paul718e3742002-12-13 20:15:29 +000097 }
98
paul020709f2003-04-04 02:44:16 +000099 ospf = ospf_lookup ();
100 if (ospf != NULL)
101 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000102
103#ifdef HAVE_SNMP
104 ospf_snmp_if_update (ifp);
105#endif /* HAVE_SNMP */
106
107 return 0;
108}
109
paul4dadc292005-05-06 21:37:42 +0000110static int
paul718e3742002-12-13 20:15:29 +0000111ospf_interface_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000112 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000113{
114 struct interface *ifp;
115 struct stream *s;
116 struct route_node *rn;
117
paulcf795c52003-06-19 02:13:25 +0000118 s = zclient->ibuf;
paul718e3742002-12-13 20:15:29 +0000119 /* zebra_interface_state_read() updates interface structure in iflist */
120 ifp = zebra_interface_state_read (s);
121
122 if (ifp == NULL)
123 return 0;
124
125 if (if_is_up (ifp))
126 zlog_warn ("Zebra: got delete of %s, but interface is still up",
paulcf795c52003-06-19 02:13:25 +0000127 ifp->name);
128
paul718e3742002-12-13 20:15:29 +0000129 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000130 zlog_debug
paulcf795c52003-06-19 02:13:25 +0000131 ("Zebra: interface delete %s index %d flags %ld metric %d mtu %d",
132 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +0000133
134#ifdef HAVE_SNMP
135 ospf_snmp_if_delete (ifp);
136#endif /* HAVE_SNMP */
137
138 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
139 if (rn->info)
140 ospf_if_free ((struct ospf_interface *) rn->info);
141
ajsd2fc8892005-04-02 18:38:43 +0000142 ifp->ifindex = IFINDEX_INTERNAL;
paul718e3742002-12-13 20:15:29 +0000143 return 0;
144}
145
ajsd2fc8892005-04-02 18:38:43 +0000146static struct interface *
paul718e3742002-12-13 20:15:29 +0000147zebra_interface_if_lookup (struct stream *s)
148{
ajs21fefa92005-04-02 23:16:41 +0000149 char ifname_tmp[INTERFACE_NAMSIZ];
paul718e3742002-12-13 20:15:29 +0000150
151 /* Read interface name. */
152 stream_get (ifname_tmp, s, INTERFACE_NAMSIZ);
153
ajsd2fc8892005-04-02 18:38:43 +0000154 /* And look it up. */
ajs21fefa92005-04-02 23:16:41 +0000155 return if_lookup_by_name_len(ifname_tmp,
156 strnlen(ifname_tmp, INTERFACE_NAMSIZ));
paul718e3742002-12-13 20:15:29 +0000157}
158
paul4dadc292005-05-06 21:37:42 +0000159static int
paul718e3742002-12-13 20:15:29 +0000160ospf_interface_state_up (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000161 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000162{
163 struct interface *ifp;
paul718e3742002-12-13 20:15:29 +0000164 struct ospf_interface *oi;
165 struct route_node *rn;
paulcf795c52003-06-19 02:13:25 +0000166
paul718e3742002-12-13 20:15:29 +0000167 ifp = zebra_interface_if_lookup (zclient->ibuf);
168
169 if (ifp == NULL)
170 return 0;
171
172 /* Interface is already up. */
paul2e3b2e42002-12-13 21:03:13 +0000173 if (if_is_operative (ifp))
paul718e3742002-12-13 20:15:29 +0000174 {
175 /* Temporarily keep ifp values. */
ajsa608bbf2005-03-29 17:03:49 +0000176 struct interface if_tmp;
paul718e3742002-12-13 20:15:29 +0000177 memcpy (&if_tmp, ifp, sizeof (struct interface));
178
179 zebra_interface_if_set_value (zclient->ibuf, ifp);
180
181 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000182 zlog_debug ("Zebra: Interface[%s] state update.", ifp->name);
paul718e3742002-12-13 20:15:29 +0000183
184 if (if_tmp.bandwidth != ifp->bandwidth)
paulcf795c52003-06-19 02:13:25 +0000185 {
186 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000187 zlog_debug ("Zebra: Interface[%s] bandwidth change %d -> %d.",
paulcf795c52003-06-19 02:13:25 +0000188 ifp->name, if_tmp.bandwidth, ifp->bandwidth);
paul718e3742002-12-13 20:15:29 +0000189
paulcf795c52003-06-19 02:13:25 +0000190 ospf_if_recalculate_output_cost (ifp);
191 }
ajsa608bbf2005-03-29 17:03:49 +0000192
193 if (if_tmp.mtu != ifp->mtu)
194 {
195 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
196 zlog_debug ("Zebra: Interface[%s] MTU change %u -> %u.",
197 ifp->name, if_tmp.mtu, ifp->mtu);
198
199 /* Must reset the interface (simulate down/up) when MTU changes. */
200 ospf_if_reset(ifp);
201 }
paul718e3742002-12-13 20:15:29 +0000202 return 0;
203 }
paulcf795c52003-06-19 02:13:25 +0000204
paul718e3742002-12-13 20:15:29 +0000205 zebra_interface_if_set_value (zclient->ibuf, ifp);
paulcf795c52003-06-19 02:13:25 +0000206
paul718e3742002-12-13 20:15:29 +0000207 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000208 zlog_debug ("Zebra: Interface[%s] state change to up.", ifp->name);
paulcf795c52003-06-19 02:13:25 +0000209
210 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +0000211 {
paulcf795c52003-06-19 02:13:25 +0000212 if ((oi = rn->info) == NULL)
213 continue;
214
paul718e3742002-12-13 20:15:29 +0000215 ospf_if_up (oi);
216 }
paulcf795c52003-06-19 02:13:25 +0000217
paul718e3742002-12-13 20:15:29 +0000218 return 0;
219}
220
paul4dadc292005-05-06 21:37:42 +0000221static int
paul718e3742002-12-13 20:15:29 +0000222ospf_interface_state_down (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000223 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000224{
225 struct interface *ifp;
226 struct ospf_interface *oi;
227 struct route_node *node;
228
229 ifp = zebra_interface_state_read (zclient->ibuf);
230
231 if (ifp == NULL)
232 return 0;
233
234 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000235 zlog_debug ("Zebra: Interface[%s] state change to down.", ifp->name);
paul718e3742002-12-13 20:15:29 +0000236
paulcf795c52003-06-19 02:13:25 +0000237 for (node = route_top (IF_OIFS (ifp)); node; node = route_next (node))
paul718e3742002-12-13 20:15:29 +0000238 {
paulcf795c52003-06-19 02:13:25 +0000239 if ((oi = node->info) == NULL)
240 continue;
paul718e3742002-12-13 20:15:29 +0000241 ospf_if_down (oi);
242 }
243
244 return 0;
245}
246
paul4dadc292005-05-06 21:37:42 +0000247static int
paul718e3742002-12-13 20:15:29 +0000248ospf_interface_address_add (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000249 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000250{
paul020709f2003-04-04 02:44:16 +0000251 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000252 struct connected *c;
253
paul0a589352004-05-08 11:48:26 +0000254 c = zebra_interface_address_read (command, zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000255
256 if (c == NULL)
257 return 0;
258
paul020709f2003-04-04 02:44:16 +0000259 ospf = ospf_lookup ();
260 if (ospf != NULL)
261 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000262
263#ifdef HAVE_SNMP
264 ospf_snmp_if_update (c->ifp);
265#endif /* HAVE_SNMP */
266
267 return 0;
268}
269
paul4dadc292005-05-06 21:37:42 +0000270static int
paul718e3742002-12-13 20:15:29 +0000271ospf_interface_address_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000272 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000273{
paul020709f2003-04-04 02:44:16 +0000274 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000275 struct connected *c;
276 struct interface *ifp;
277 struct ospf_interface *oi;
278 struct route_node *rn;
279 struct prefix p;
280
paul0a589352004-05-08 11:48:26 +0000281 c = zebra_interface_address_read (command, zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000282
283 if (c == NULL)
284 return 0;
285
286 ifp = c->ifp;
287 p = *c->address;
288 p.prefixlen = IPV4_MAX_PREFIXLEN;
289
290 rn = route_node_lookup (IF_OIFS (ifp), &p);
paulcf795c52003-06-19 02:13:25 +0000291 if (!rn)
paul98429f62006-01-10 22:11:54 +0000292 {
293 connected_free (c);
294 return 0;
295 }
paul718e3742002-12-13 20:15:29 +0000296
297 assert (rn->info);
298 oi = rn->info;
paulcf795c52003-06-19 02:13:25 +0000299
paul718e3742002-12-13 20:15:29 +0000300 /* Call interface hook functions to clean up */
301 ospf_if_free (oi);
paulcf795c52003-06-19 02:13:25 +0000302
paul718e3742002-12-13 20:15:29 +0000303#ifdef HAVE_SNMP
304 ospf_snmp_if_update (c->ifp);
305#endif /* HAVE_SNMP */
306
307 connected_free (c);
308
paul020709f2003-04-04 02:44:16 +0000309 ospf = ospf_lookup ();
310 if (ospf != NULL)
311 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000312
313 return 0;
314}
paul72357f22003-06-19 02:11:23 +0000315
paul718e3742002-12-13 20:15:29 +0000316void
317ospf_zebra_add (struct prefix_ipv4 *p, struct ospf_route *or)
318{
319 u_char message;
320 u_char distance;
321 u_char flags;
322 int psize;
323 struct stream *s;
324 struct ospf_path *path;
hasso52dc7ee2004-09-23 19:18:23 +0000325 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000326
327 if (zclient->redist[ZEBRA_ROUTE_OSPF])
328 {
329 message = 0;
330 flags = 0;
331
332 /* OSPF pass nexthop and metric */
333 SET_FLAG (message, ZAPI_MESSAGE_NEXTHOP);
334 SET_FLAG (message, ZAPI_MESSAGE_METRIC);
335
336 /* Distance value. */
337 distance = ospf_distance_apply (p, or);
338 if (distance)
paul72357f22003-06-19 02:11:23 +0000339 SET_FLAG (message, ZAPI_MESSAGE_DISTANCE);
paul718e3742002-12-13 20:15:29 +0000340
341 /* Make packet. */
342 s = zclient->obuf;
343 stream_reset (s);
344
paul718e3742002-12-13 20:15:29 +0000345 /* Put command, type, flags, message. */
paulc6371712006-01-17 17:49:53 +0000346 zclient_create_header (s, ZEBRA_IPV4_ROUTE_ADD);
paul718e3742002-12-13 20:15:29 +0000347 stream_putc (s, ZEBRA_ROUTE_OSPF);
348 stream_putc (s, flags);
349 stream_putc (s, message);
paulcf795c52003-06-19 02:13:25 +0000350
paul718e3742002-12-13 20:15:29 +0000351 /* Put prefix information. */
352 psize = PSIZE (p->prefixlen);
353 stream_putc (s, p->prefixlen);
paulcf795c52003-06-19 02:13:25 +0000354 stream_write (s, (u_char *) & p->prefix, psize);
paul718e3742002-12-13 20:15:29 +0000355
356 /* Nexthop count. */
paul96735ee2003-08-10 02:51:22 +0000357 stream_putc (s, or->paths->count);
paul718e3742002-12-13 20:15:29 +0000358
359 /* Nexthop, ifindex, distance and metric information. */
paul1eb8ef22005-04-07 07:30:20 +0000360 for (ALL_LIST_ELEMENTS_RO (or->paths, node, path))
paul72357f22003-06-19 02:11:23 +0000361 {
paulcf795c52003-06-19 02:13:25 +0000362 if (path->nexthop.s_addr != INADDR_ANY)
363 {
364 stream_putc (s, ZEBRA_NEXTHOP_IPV4);
365 stream_put_in_addr (s, &path->nexthop);
366 }
367 else
368 {
369 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
370 if (path->oi)
371 stream_putl (s, path->oi->ifp->ifindex);
372 else
373 stream_putl (s, 0);
374 }
paul72357f22003-06-19 02:11:23 +0000375
376 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
377 {
ajs9b0e25c2004-12-08 19:06:51 +0000378 zlog_debug ("Zebra: Route add %s/%d nexthop %s",
paulcf795c52003-06-19 02:13:25 +0000379 inet_ntoa (p->prefix),
380 p->prefixlen, inet_ntoa (path->nexthop));
381 }
382 }
paul718e3742002-12-13 20:15:29 +0000383
384 if (CHECK_FLAG (message, ZAPI_MESSAGE_DISTANCE))
paulcf795c52003-06-19 02:13:25 +0000385 stream_putc (s, distance);
paul718e3742002-12-13 20:15:29 +0000386 if (CHECK_FLAG (message, ZAPI_MESSAGE_METRIC))
paulcf795c52003-06-19 02:13:25 +0000387 {
388 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL)
389 stream_putl (s, or->cost + or->u.ext.type2_cost);
390 else if (or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
391 stream_putl (s, or->u.ext.type2_cost);
392 else
393 stream_putl (s, or->cost);
394 }
paul718e3742002-12-13 20:15:29 +0000395
396 stream_putw_at (s, 0, stream_get_endp (s));
397
ajs634f9ea2005-04-11 15:51:40 +0000398 zclient_send_message(zclient);
paul718e3742002-12-13 20:15:29 +0000399 }
400}
401
402void
403ospf_zebra_delete (struct prefix_ipv4 *p, struct ospf_route *or)
404{
405 struct zapi_ipv4 api;
paul72357f22003-06-19 02:11:23 +0000406 struct ospf_path *path;
407 struct in_addr *nexthop;
paul1eb8ef22005-04-07 07:30:20 +0000408 struct listnode *node, *nnode;
paul718e3742002-12-13 20:15:29 +0000409
410 if (zclient->redist[ZEBRA_ROUTE_OSPF])
411 {
412 api.type = ZEBRA_ROUTE_OSPF;
413 api.flags = 0;
414 api.message = 0;
paul72357f22003-06-19 02:11:23 +0000415 api.ifindex_num = 0;
416 api.nexthop_num = 0;
paul718e3742002-12-13 20:15:29 +0000417
paul1eb8ef22005-04-07 07:30:20 +0000418 for (ALL_LIST_ELEMENTS (or->paths, node, nnode, path))
paulcf795c52003-06-19 02:13:25 +0000419 {
paulcf795c52003-06-19 02:13:25 +0000420 if (path->nexthop.s_addr != INADDR_ANY)
421 {
pauld8e1d6b2003-08-10 04:04:41 +0000422 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
paulcf795c52003-06-19 02:13:25 +0000423 api.nexthop_num = 1;
424 nexthop = &path->nexthop;
425 api.nexthop = &nexthop;
426 }
hasso2db3d052004-02-11 21:52:13 +0000427 else if (ospf_if_exists(path->oi) && (path->oi->ifp))
paulbb8ff1e2003-08-12 06:00:30 +0000428 {
429 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
430 api.ifindex_num = 1;
431 api.ifindex = &path->oi->ifp->ifindex;
432 }
433 else if ( IS_DEBUG_OSPF(zebra,ZEBRA_REDISTRIBUTE) )
434 {
ajs9b0e25c2004-12-08 19:06:51 +0000435 zlog_debug("Zebra: no ifp %s %d",
paulbb8ff1e2003-08-12 06:00:30 +0000436 inet_ntoa(p->prefix),
437 p->prefixlen);
438 }
paul72357f22003-06-19 02:11:23 +0000439
paul0a589352004-05-08 11:48:26 +0000440 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
paul72357f22003-06-19 02:11:23 +0000441
442 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.nexthop_num)
paulcf795c52003-06-19 02:13:25 +0000443 {
ajs9b0e25c2004-12-08 19:06:51 +0000444 zlog_debug ("Zebra: Route delete %s/%d nexthop %s",
paulcf795c52003-06-19 02:13:25 +0000445 inet_ntoa (p->prefix),
446 p->prefixlen, inet_ntoa (**api.nexthop));
447 }
paulbb8ff1e2003-08-12 06:00:30 +0000448 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.ifindex_num)
449 {
ajs9b0e25c2004-12-08 19:06:51 +0000450 zlog_debug ("Zebra: Route delete %s/%d ifindex %d",
paulbb8ff1e2003-08-12 06:00:30 +0000451 inet_ntoa (p->prefix),
452 p->prefixlen, *api.ifindex);
453 }
paulcf795c52003-06-19 02:13:25 +0000454 }
paul718e3742002-12-13 20:15:29 +0000455 }
456}
457
458void
459ospf_zebra_add_discard (struct prefix_ipv4 *p)
460{
461 struct zapi_ipv4 api;
462
463 if (zclient->redist[ZEBRA_ROUTE_OSPF])
464 {
465 api.type = ZEBRA_ROUTE_OSPF;
466 api.flags = ZEBRA_FLAG_BLACKHOLE;
467 api.message = 0;
468 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
469 api.nexthop_num = 0;
470 api.ifindex_num = 0;
471
paul0a589352004-05-08 11:48:26 +0000472 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_ADD, zclient, p, &api);
paul718e3742002-12-13 20:15:29 +0000473 }
474}
475
476void
477ospf_zebra_delete_discard (struct prefix_ipv4 *p)
478{
479 struct zapi_ipv4 api;
480
481 if (zclient->redist[ZEBRA_ROUTE_OSPF])
482 {
483 api.type = ZEBRA_ROUTE_OSPF;
484 api.flags = ZEBRA_FLAG_BLACKHOLE;
485 api.message = 0;
486 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
487 api.nexthop_num = 0;
488 api.ifindex_num = 0;
489
paul0a589352004-05-08 11:48:26 +0000490 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
paul72357f22003-06-19 02:11:23 +0000491
492 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000493 zlog_debug ("Zebra: Route delete discard %s/%d",
paulcf795c52003-06-19 02:13:25 +0000494 inet_ntoa (p->prefix), p->prefixlen);
paul72357f22003-06-19 02:11:23 +0000495
paul718e3742002-12-13 20:15:29 +0000496 }
497}
498
499int
500ospf_is_type_redistributed (int type)
501{
502 return (DEFAULT_ROUTE_TYPE (type)) ?
503 zclient->default_information : zclient->redist[type];
504}
505
506int
paul020709f2003-04-04 02:44:16 +0000507ospf_redistribute_set (struct ospf *ospf, int type, int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000508{
509 int force = 0;
paulcf795c52003-06-19 02:13:25 +0000510
paul718e3742002-12-13 20:15:29 +0000511 if (ospf_is_type_redistributed (type))
512 {
paul68980082003-03-25 05:07:42 +0000513 if (mtype != ospf->dmetric[type].type)
paulcf795c52003-06-19 02:13:25 +0000514 {
515 ospf->dmetric[type].type = mtype;
516 force = LSA_REFRESH_FORCE;
517 }
paul68980082003-03-25 05:07:42 +0000518 if (mvalue != ospf->dmetric[type].value)
paulcf795c52003-06-19 02:13:25 +0000519 {
520 ospf->dmetric[type].value = mvalue;
521 force = LSA_REFRESH_FORCE;
522 }
523
paul68980082003-03-25 05:07:42 +0000524 ospf_external_lsa_refresh_type (ospf, type, force);
paulcf795c52003-06-19 02:13:25 +0000525
paul718e3742002-12-13 20:15:29 +0000526 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000527 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000528 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000529 metric_type (ospf, type), metric_value (ospf, type));
530
paul718e3742002-12-13 20:15:29 +0000531 return CMD_SUCCESS;
532 }
533
paul68980082003-03-25 05:07:42 +0000534 ospf->dmetric[type].type = mtype;
535 ospf->dmetric[type].value = mvalue;
paul718e3742002-12-13 20:15:29 +0000536
paul0a589352004-05-08 11:48:26 +0000537 zclient_redistribute (ZEBRA_REDISTRIBUTE_ADD, zclient, type);
paul718e3742002-12-13 20:15:29 +0000538
539 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000540 zlog_debug ("Redistribute[%s]: Start Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000541 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000542 metric_type (ospf, type), metric_value (ospf, type));
543
paul68980082003-03-25 05:07:42 +0000544 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000545
546 return CMD_SUCCESS;
547}
548
549int
paul020709f2003-04-04 02:44:16 +0000550ospf_redistribute_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000551{
552 if (type == zclient->redist_default)
553 return CMD_SUCCESS;
554
paulcf795c52003-06-19 02:13:25 +0000555 if (!ospf_is_type_redistributed (type))
paul718e3742002-12-13 20:15:29 +0000556 return CMD_SUCCESS;
557
paul0a589352004-05-08 11:48:26 +0000558 zclient_redistribute (ZEBRA_REDISTRIBUTE_DELETE, zclient, type);
paulcf795c52003-06-19 02:13:25 +0000559
paul718e3742002-12-13 20:15:29 +0000560 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000561 zlog_debug ("Redistribute[%s]: Stop",
ajsf52d13c2005-10-01 17:38:06 +0000562 ospf_redist_string(type));
paul718e3742002-12-13 20:15:29 +0000563
paul68980082003-03-25 05:07:42 +0000564 ospf->dmetric[type].type = -1;
565 ospf->dmetric[type].value = -1;
paul718e3742002-12-13 20:15:29 +0000566
567 /* Remove the routes from OSPF table. */
568 ospf_redistribute_withdraw (type);
569
paul68980082003-03-25 05:07:42 +0000570 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000571
572 return CMD_SUCCESS;
573}
574
575int
paul020709f2003-04-04 02:44:16 +0000576ospf_redistribute_default_set (struct ospf *ospf, int originate,
paulcf795c52003-06-19 02:13:25 +0000577 int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000578{
579 int force = 0;
paul020709f2003-04-04 02:44:16 +0000580
paul718e3742002-12-13 20:15:29 +0000581 if (ospf_is_type_redistributed (DEFAULT_ROUTE))
582 {
paul68980082003-03-25 05:07:42 +0000583 if (mtype != ospf->dmetric[DEFAULT_ROUTE].type)
paulcf795c52003-06-19 02:13:25 +0000584 {
585 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
586 force = 1;
587 }
paul68980082003-03-25 05:07:42 +0000588 if (mvalue != ospf->dmetric[DEFAULT_ROUTE].value)
paulcf795c52003-06-19 02:13:25 +0000589 {
590 force = 1;
591 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
592 }
593
paul68980082003-03-25 05:07:42 +0000594 ospf_external_lsa_refresh_default (ospf);
paulcf795c52003-06-19 02:13:25 +0000595
paul718e3742002-12-13 20:15:29 +0000596 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000597 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
ajsf52d13c2005-10-01 17:38:06 +0000598 ospf_redist_string(DEFAULT_ROUTE),
paulcf795c52003-06-19 02:13:25 +0000599 metric_type (ospf, DEFAULT_ROUTE),
600 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000601 return CMD_SUCCESS;
602 }
603
paul68980082003-03-25 05:07:42 +0000604 ospf->default_originate = originate;
605 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
606 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
paul718e3742002-12-13 20:15:29 +0000607
paul0a589352004-05-08 11:48:26 +0000608 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_ADD, zclient);
paulcf795c52003-06-19 02:13:25 +0000609
paul718e3742002-12-13 20:15:29 +0000610 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000611 zlog_debug ("Redistribute[DEFAULT]: Start Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000612 metric_type (ospf, DEFAULT_ROUTE),
613 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000614
paul68980082003-03-25 05:07:42 +0000615 if (ospf->router_id.s_addr == 0)
616 ospf->external_origin |= (1 << DEFAULT_ROUTE);
paul718e3742002-12-13 20:15:29 +0000617 else
Paul Jakma4021b602006-05-12 22:55:41 +0000618 thread_add_timer (master, ospf_default_originate_timer, ospf, 1);
paul718e3742002-12-13 20:15:29 +0000619
paul68980082003-03-25 05:07:42 +0000620 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000621
622 return CMD_SUCCESS;
623}
624
625int
paul020709f2003-04-04 02:44:16 +0000626ospf_redistribute_default_unset (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +0000627{
628 if (!ospf_is_type_redistributed (DEFAULT_ROUTE))
629 return CMD_SUCCESS;
630
paul68980082003-03-25 05:07:42 +0000631 ospf->default_originate = DEFAULT_ORIGINATE_NONE;
632 ospf->dmetric[DEFAULT_ROUTE].type = -1;
633 ospf->dmetric[DEFAULT_ROUTE].value = -1;
paul718e3742002-12-13 20:15:29 +0000634
paul0a589352004-05-08 11:48:26 +0000635 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_DELETE, zclient);
paul718e3742002-12-13 20:15:29 +0000636
637 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000638 zlog_debug ("Redistribute[DEFAULT]: Stop");
paulcf795c52003-06-19 02:13:25 +0000639
paul68980082003-03-25 05:07:42 +0000640 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000641
642 return CMD_SUCCESS;
643}
644
paul4dadc292005-05-06 21:37:42 +0000645static int
paul020709f2003-04-04 02:44:16 +0000646ospf_external_lsa_originate_check (struct ospf *ospf,
paulcf795c52003-06-19 02:13:25 +0000647 struct external_info *ei)
paul718e3742002-12-13 20:15:29 +0000648{
649 /* If prefix is multicast, then do not originate LSA. */
650 if (IN_MULTICAST (htonl (ei->p.prefix.s_addr)))
651 {
652 zlog_info ("LSA[Type5:%s]: Not originate AS-external-LSA, "
paulcf795c52003-06-19 02:13:25 +0000653 "Prefix belongs multicast", inet_ntoa (ei->p.prefix));
paul718e3742002-12-13 20:15:29 +0000654 return 0;
655 }
656
657 /* Take care of default-originate. */
658 if (is_prefix_default (&ei->p))
paul68980082003-03-25 05:07:42 +0000659 if (ospf->default_originate == DEFAULT_ORIGINATE_NONE)
paul718e3742002-12-13 20:15:29 +0000660 {
paulcf795c52003-06-19 02:13:25 +0000661 zlog_info ("LSA[Type5:0.0.0.0]: Not originate AS-exntenal-LSA "
662 "for default");
663 return 0;
paul718e3742002-12-13 20:15:29 +0000664 }
665
666 return 1;
667}
668
669/* If connected prefix is OSPF enable interface, then do not announce. */
670int
paulcf795c52003-06-19 02:13:25 +0000671ospf_distribute_check_connected (struct ospf *ospf, struct external_info *ei)
paul718e3742002-12-13 20:15:29 +0000672{
673 struct route_node *rn;
674
paul68980082003-03-25 05:07:42 +0000675 for (rn = route_top (ospf->networks); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +0000676 if (rn->info != NULL)
paulcf795c52003-06-19 02:13:25 +0000677 if (prefix_match (&rn->p, (struct prefix *) &ei->p))
678 {
679 route_unlock_node (rn);
680 return 0;
681 }
paul718e3742002-12-13 20:15:29 +0000682
683 return 1;
684}
685
686/* return 1 if external LSA must be originated, 0 otherwise */
687int
paul68980082003-03-25 05:07:42 +0000688ospf_redistribute_check (struct ospf *ospf,
paulcf795c52003-06-19 02:13:25 +0000689 struct external_info *ei, int *changed)
paul718e3742002-12-13 20:15:29 +0000690{
691 struct route_map_set_values save_values;
692 struct prefix_ipv4 *p = &ei->p;
693 u_char type = is_prefix_default (&ei->p) ? DEFAULT_ROUTE : ei->type;
paulcf795c52003-06-19 02:13:25 +0000694
paul718e3742002-12-13 20:15:29 +0000695 if (changed)
696 *changed = 0;
697
paul020709f2003-04-04 02:44:16 +0000698 if (!ospf_external_lsa_originate_check (ospf, ei))
paul718e3742002-12-13 20:15:29 +0000699 return 0;
700
701 /* Take care connected route. */
paul68980082003-03-25 05:07:42 +0000702 if (type == ZEBRA_ROUTE_CONNECT &&
703 !ospf_distribute_check_connected (ospf, ei))
paul718e3742002-12-13 20:15:29 +0000704 return 0;
705
paul020709f2003-04-04 02:44:16 +0000706 if (!DEFAULT_ROUTE_TYPE (type) && DISTRIBUTE_NAME (ospf, type))
paul718e3742002-12-13 20:15:29 +0000707 /* distirbute-list exists, but access-list may not? */
paul020709f2003-04-04 02:44:16 +0000708 if (DISTRIBUTE_LIST (ospf, type))
709 if (access_list_apply (DISTRIBUTE_LIST (ospf, type), p) == FILTER_DENY)
paulcf795c52003-06-19 02:13:25 +0000710 {
711 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000712 zlog_debug ("Redistribute[%s]: %s/%d filtered by ditribute-list.",
ajsf52d13c2005-10-01 17:38:06 +0000713 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000714 inet_ntoa (p->prefix), p->prefixlen);
715 return 0;
716 }
paul718e3742002-12-13 20:15:29 +0000717
718 save_values = ei->route_map_set;
719 ospf_reset_route_map_set_values (&ei->route_map_set);
paulcf795c52003-06-19 02:13:25 +0000720
paul718e3742002-12-13 20:15:29 +0000721 /* apply route-map if needed */
paul020709f2003-04-04 02:44:16 +0000722 if (ROUTEMAP_NAME (ospf, type))
paul718e3742002-12-13 20:15:29 +0000723 {
724 int ret;
725
paulcf795c52003-06-19 02:13:25 +0000726 ret = route_map_apply (ROUTEMAP (ospf, type), (struct prefix *) p,
727 RMAP_OSPF, ei);
paul718e3742002-12-13 20:15:29 +0000728
729 if (ret == RMAP_DENYMATCH)
paulcf795c52003-06-19 02:13:25 +0000730 {
731 ei->route_map_set = save_values;
732 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000733 zlog_debug ("Redistribute[%s]: %s/%d filtered by route-map.",
ajsf52d13c2005-10-01 17:38:06 +0000734 ospf_redist_string(type),
paulcf795c52003-06-19 02:13:25 +0000735 inet_ntoa (p->prefix), p->prefixlen);
736 return 0;
737 }
738
paul718e3742002-12-13 20:15:29 +0000739 /* check if 'route-map set' changed something */
740 if (changed)
paulcf795c52003-06-19 02:13:25 +0000741 *changed = !ospf_route_map_set_compare (&ei->route_map_set,
742 &save_values);
paul718e3742002-12-13 20:15:29 +0000743 }
744
745 return 1;
746}
747
748/* OSPF route-map set for redistribution */
749void
paul6c835672004-10-11 11:00:30 +0000750ospf_routemap_set (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000751{
paul020709f2003-04-04 02:44:16 +0000752 if (ROUTEMAP_NAME (ospf, type))
753 free (ROUTEMAP_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000754
paul020709f2003-04-04 02:44:16 +0000755 ROUTEMAP_NAME (ospf, type) = strdup (name);
756 ROUTEMAP (ospf, type) = route_map_lookup_by_name (name);
paul718e3742002-12-13 20:15:29 +0000757}
758
759void
paul020709f2003-04-04 02:44:16 +0000760ospf_routemap_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000761{
paul020709f2003-04-04 02:44:16 +0000762 if (ROUTEMAP_NAME (ospf, type))
763 free (ROUTEMAP_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000764
paul020709f2003-04-04 02:44:16 +0000765 ROUTEMAP_NAME (ospf, type) = NULL;
766 ROUTEMAP (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000767}
768
769/* Zebra route add and delete treatment. */
paul4dadc292005-05-06 21:37:42 +0000770static int
paul718e3742002-12-13 20:15:29 +0000771ospf_zebra_read_ipv4 (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000772 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000773{
774 struct stream *s;
775 struct zapi_ipv4 api;
776 unsigned long ifindex;
777 struct in_addr nexthop;
778 struct prefix_ipv4 p;
779 struct external_info *ei;
paul020709f2003-04-04 02:44:16 +0000780 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000781
782 s = zclient->ibuf;
783 ifindex = 0;
784 nexthop.s_addr = 0;
785
786 /* Type, flags, message. */
787 api.type = stream_getc (s);
788 api.flags = stream_getc (s);
789 api.message = stream_getc (s);
790
791 /* IPv4 prefix. */
792 memset (&p, 0, sizeof (struct prefix_ipv4));
793 p.family = AF_INET;
794 p.prefixlen = stream_getc (s);
795 stream_get (&p.prefix, s, PSIZE (p.prefixlen));
796
hasso8585d4e2004-04-20 17:25:12 +0000797 if (IPV4_NET127(ntohl(p.prefix.s_addr)))
798 return 0;
799
paul718e3742002-12-13 20:15:29 +0000800 /* Nexthop, ifindex, distance, metric. */
801 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP))
802 {
803 api.nexthop_num = stream_getc (s);
804 nexthop.s_addr = stream_get_ipv4 (s);
805 }
806 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_IFINDEX))
807 {
808 api.ifindex_num = stream_getc (s);
gdt6a8da852004-02-13 17:40:51 +0000809 /* XXX assert(api.ifindex_num == 1); */
paul718e3742002-12-13 20:15:29 +0000810 ifindex = stream_getl (s);
811 }
812 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_DISTANCE))
813 api.distance = stream_getc (s);
814 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_METRIC))
815 api.metric = stream_getl (s);
816
paul020709f2003-04-04 02:44:16 +0000817 ospf = ospf_lookup ();
818 if (ospf == NULL)
819 return 0;
820
paul718e3742002-12-13 20:15:29 +0000821 if (command == ZEBRA_IPV4_ROUTE_ADD)
822 {
paul7021c422003-07-15 12:52:22 +0000823 /* XXX|HACK|TODO|FIXME:
hassoa0a39762004-04-23 08:51:10 +0000824 * Maybe we should ignore reject/blackhole routes? Testing shows that
825 * there is no problems though and this is only way to "summarize"
826 * routes in ASBR at the moment. Maybe we need just a better generalised
827 * solution for these types?
828 *
829 * if ( CHECK_FLAG (api.flags, ZEBRA_FLAG_BLACKHOLE)
830 * || CHECK_FLAG (api.flags, ZEBRA_FLAG_REJECT))
831 * return 0;
paul7021c422003-07-15 12:52:22 +0000832 */
paul7021c422003-07-15 12:52:22 +0000833
paul718e3742002-12-13 20:15:29 +0000834 ei = ospf_external_info_add (api.type, p, ifindex, nexthop);
835
paul68980082003-03-25 05:07:42 +0000836 if (ospf->router_id.s_addr == 0)
paulcf795c52003-06-19 02:13:25 +0000837 /* Set flags to generate AS-external-LSA originate event
838 for each redistributed protocols later. */
839 ospf->external_origin |= (1 << api.type);
paul718e3742002-12-13 20:15:29 +0000840 else
paulcf795c52003-06-19 02:13:25 +0000841 {
842 if (ei)
843 {
844 if (is_prefix_default (&p))
845 ospf_external_lsa_refresh_default (ospf);
846 else
847 {
848 struct ospf_lsa *current;
paul718e3742002-12-13 20:15:29 +0000849
paulcf795c52003-06-19 02:13:25 +0000850 current = ospf_external_info_find_lsa (ospf, &ei->p);
851 if (!current)
852 ospf_external_lsa_originate (ospf, ei);
853 else if (IS_LSA_MAXAGE (current))
854 ospf_external_lsa_refresh (ospf, current,
855 ei, LSA_REFRESH_FORCE);
856 else
857 zlog_warn ("ospf_zebra_read_ipv4() : %s already exists",
858 inet_ntoa (p.prefix));
859 }
860 }
861 }
paul718e3742002-12-13 20:15:29 +0000862 }
paulcf795c52003-06-19 02:13:25 +0000863 else /* if (command == ZEBRA_IPV4_ROUTE_DELETE) */
paul718e3742002-12-13 20:15:29 +0000864 {
865 ospf_external_info_delete (api.type, p);
paul7021c422003-07-15 12:52:22 +0000866 if (is_prefix_default (&p))
paulcf795c52003-06-19 02:13:25 +0000867 ospf_external_lsa_refresh_default (ospf);
paul7021c422003-07-15 12:52:22 +0000868 else
ajs5339cfd2005-09-19 13:28:05 +0000869 ospf_external_lsa_flush (ospf, api.type, &p, ifindex /*, nexthop */);
paul718e3742002-12-13 20:15:29 +0000870 }
871
872 return 0;
873}
paul718e3742002-12-13 20:15:29 +0000874
paulcf795c52003-06-19 02:13:25 +0000875
paul718e3742002-12-13 20:15:29 +0000876int
paul6c835672004-10-11 11:00:30 +0000877ospf_distribute_list_out_set (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000878{
879 /* Lookup access-list for distribute-list. */
paul020709f2003-04-04 02:44:16 +0000880 DISTRIBUTE_LIST (ospf, type) = access_list_lookup (AFI_IP, name);
paul718e3742002-12-13 20:15:29 +0000881
882 /* Clear previous distribute-name. */
paul020709f2003-04-04 02:44:16 +0000883 if (DISTRIBUTE_NAME (ospf, type))
884 free (DISTRIBUTE_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000885
886 /* Set distribute-name. */
paul020709f2003-04-04 02:44:16 +0000887 DISTRIBUTE_NAME (ospf, type) = strdup (name);
paul718e3742002-12-13 20:15:29 +0000888
889 /* If access-list have been set, schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000890 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000891 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000892
893 return CMD_SUCCESS;
894}
895
896int
paul6c835672004-10-11 11:00:30 +0000897ospf_distribute_list_out_unset (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000898{
899 /* Schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000900 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000901 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000902
903 /* Unset distribute-list. */
paul020709f2003-04-04 02:44:16 +0000904 DISTRIBUTE_LIST (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000905
906 /* Clear distribute-name. */
paul020709f2003-04-04 02:44:16 +0000907 if (DISTRIBUTE_NAME (ospf, type))
908 free (DISTRIBUTE_NAME (ospf, type));
paulcf795c52003-06-19 02:13:25 +0000909
paul020709f2003-04-04 02:44:16 +0000910 DISTRIBUTE_NAME (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000911
912 return CMD_SUCCESS;
913}
914
915/* distribute-list update timer. */
paul4dadc292005-05-06 21:37:42 +0000916static int
paul718e3742002-12-13 20:15:29 +0000917ospf_distribute_list_update_timer (struct thread *thread)
918{
919 struct route_node *rn;
920 struct external_info *ei;
921 struct route_table *rt;
922 struct ospf_lsa *lsa;
paul4dadc292005-05-06 21:37:42 +0000923 intptr_t type;
paul020709f2003-04-04 02:44:16 +0000924 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000925
paul4dadc292005-05-06 21:37:42 +0000926 type = (intptr_t)THREAD_ARG (thread);
hasso01018ce2005-08-05 07:40:15 +0000927 assert (type <= ZEBRA_ROUTE_MAX);
paul64511f32004-10-31 18:01:13 +0000928
paul718e3742002-12-13 20:15:29 +0000929 rt = EXTERNAL_INFO (type);
930
paul020709f2003-04-04 02:44:16 +0000931 ospf = ospf_lookup ();
932 if (ospf == NULL)
933 return 0;
934
paul68980082003-03-25 05:07:42 +0000935 ospf->t_distribute_update = NULL;
paul718e3742002-12-13 20:15:29 +0000936
937 zlog_info ("Zebra[Redistribute]: distribute-list update timer fired!");
938
939 /* foreach all external info. */
940 if (rt)
941 for (rn = route_top (rt); rn; rn = route_next (rn))
942 if ((ei = rn->info) != NULL)
paulcf795c52003-06-19 02:13:25 +0000943 {
944 if (is_prefix_default (&ei->p))
945 ospf_external_lsa_refresh_default (ospf);
946 else if ((lsa = ospf_external_info_find_lsa (ospf, &ei->p)))
947 ospf_external_lsa_refresh (ospf, lsa, ei, LSA_REFRESH_IF_CHANGED);
948 else
949 ospf_external_lsa_originate (ospf, ei);
950 }
paul718e3742002-12-13 20:15:29 +0000951 return 0;
952}
953
954#define OSPF_DISTRIBUTE_UPDATE_DELAY 5
955
956/* Update distribute-list and set timer to apply access-list. */
957void
paul68980082003-03-25 05:07:42 +0000958ospf_distribute_list_update (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000959{
960 struct route_table *rt;
paulcf795c52003-06-19 02:13:25 +0000961
paul718e3742002-12-13 20:15:29 +0000962 /* External info does not exist. */
963 if (!(rt = EXTERNAL_INFO (type)))
964 return;
965
966 /* If exists previously invoked thread, then cancel it. */
paul68980082003-03-25 05:07:42 +0000967 if (ospf->t_distribute_update)
968 OSPF_TIMER_OFF (ospf->t_distribute_update);
paul718e3742002-12-13 20:15:29 +0000969
970 /* Set timer. */
paul68980082003-03-25 05:07:42 +0000971 ospf->t_distribute_update =
paul718e3742002-12-13 20:15:29 +0000972 thread_add_timer (master, ospf_distribute_list_update_timer,
paulcf795c52003-06-19 02:13:25 +0000973 (void *) type, OSPF_DISTRIBUTE_UPDATE_DELAY);
paul718e3742002-12-13 20:15:29 +0000974}
975
976/* If access-list is updated, apply some check. */
paul4dadc292005-05-06 21:37:42 +0000977static void
paul718e3742002-12-13 20:15:29 +0000978ospf_filter_update (struct access_list *access)
979{
paul020709f2003-04-04 02:44:16 +0000980 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000981 int type;
982 int abr_inv = 0;
983 struct ospf_area *area;
hasso52dc7ee2004-09-23 19:18:23 +0000984 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000985
986 /* If OSPF instatnce does not exist, return right now. */
paul020709f2003-04-04 02:44:16 +0000987 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +0000988 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +0000989 return;
990
paul718e3742002-12-13 20:15:29 +0000991 /* Update distribute-list, and apply filter. */
hasso01018ce2005-08-05 07:40:15 +0000992 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
paul718e3742002-12-13 20:15:29 +0000993 {
paul020709f2003-04-04 02:44:16 +0000994 if (ROUTEMAP (ospf, type) != NULL)
paulcf795c52003-06-19 02:13:25 +0000995 {
996 /* if route-map is not NULL it may be using this access list */
997 ospf_distribute_list_update (ospf, type);
998 continue;
999 }
1000
hasso01018ce2005-08-05 07:40:15 +00001001 /* There is place for route-map for default-information (ZEBRA_ROUTE_MAX),
1002 * but no distribute list. */
1003 if (type == ZEBRA_ROUTE_MAX)
1004 break;
paul718e3742002-12-13 20:15:29 +00001005
paul020709f2003-04-04 02:44:16 +00001006 if (DISTRIBUTE_NAME (ospf, type))
paulcf795c52003-06-19 02:13:25 +00001007 {
1008 /* Keep old access-list for distribute-list. */
1009 struct access_list *old = DISTRIBUTE_LIST (ospf, type);
1010
1011 /* Update access-list for distribute-list. */
1012 DISTRIBUTE_LIST (ospf, type) =
1013 access_list_lookup (AFI_IP, DISTRIBUTE_NAME (ospf, type));
1014
1015 /* No update for this distribute type. */
1016 if (old == NULL && DISTRIBUTE_LIST (ospf, type) == NULL)
1017 continue;
1018
1019 /* Schedule distribute-list update timer. */
1020 if (DISTRIBUTE_LIST (ospf, type) == NULL ||
1021 strcmp (DISTRIBUTE_NAME (ospf, type), access->name) == 0)
1022 ospf_distribute_list_update (ospf, type);
1023 }
paul718e3742002-12-13 20:15:29 +00001024 }
1025
1026 /* Update Area access-list. */
paul1eb8ef22005-04-07 07:30:20 +00001027 for (ALL_LIST_ELEMENTS_RO (ospf->areas, node, area))
1028 {
1029 if (EXPORT_NAME (area))
1030 {
1031 EXPORT_LIST (area) = NULL;
1032 abr_inv++;
1033 }
paul718e3742002-12-13 20:15:29 +00001034
paul1eb8ef22005-04-07 07:30:20 +00001035 if (IMPORT_NAME (area))
1036 {
1037 IMPORT_LIST (area) = NULL;
1038 abr_inv++;
1039 }
1040 }
paul718e3742002-12-13 20:15:29 +00001041
1042 /* Schedule ABR tasks -- this will be changed -- takada. */
paul020709f2003-04-04 02:44:16 +00001043 if (IS_OSPF_ABR (ospf) && abr_inv)
paul68980082003-03-25 05:07:42 +00001044 ospf_schedule_abr_task (ospf);
paul718e3742002-12-13 20:15:29 +00001045}
hassodd669bb2004-05-10 07:43:59 +00001046
1047/* If prefix-list is updated, do some updates. */
1048void
1049ospf_prefix_list_update (struct prefix_list *plist)
1050{
1051 struct ospf *ospf;
1052 int type;
1053 int abr_inv = 0;
1054 struct ospf_area *area;
hasso52dc7ee2004-09-23 19:18:23 +00001055 struct listnode *node;
hassodd669bb2004-05-10 07:43:59 +00001056
1057 /* If OSPF instatnce does not exist, return right now. */
1058 ospf = ospf_lookup ();
1059 if (ospf == NULL)
1060 return;
1061
1062 /* Update all route-maps which are used as redistribution filters.
1063 * They might use prefix-list.
1064 */
hasso01018ce2005-08-05 07:40:15 +00001065 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
hassodd669bb2004-05-10 07:43:59 +00001066 {
1067 if (ROUTEMAP (ospf, type) != NULL)
1068 {
1069 /* If route-map is not NULL it may be using this prefix list */
1070 ospf_distribute_list_update (ospf, type);
1071 continue;
1072 }
1073 }
1074
1075 /* Update area filter-lists. */
paul1eb8ef22005-04-07 07:30:20 +00001076 for (ALL_LIST_ELEMENTS_RO (ospf->areas, node, area))
1077 {
1078 /* Update filter-list in. */
1079 if (PREFIX_NAME_IN (area))
1080 if (strcmp (PREFIX_NAME_IN (area), plist->name) == 0)
1081 {
1082 PREFIX_LIST_IN (area) =
1083 prefix_list_lookup (AFI_IP, PREFIX_NAME_IN (area));
1084 abr_inv++;
1085 }
hassodd669bb2004-05-10 07:43:59 +00001086
paul1eb8ef22005-04-07 07:30:20 +00001087 /* Update filter-list out. */
1088 if (PREFIX_NAME_OUT (area))
1089 if (strcmp (PREFIX_NAME_OUT (area), plist->name) == 0)
1090 {
1091 PREFIX_LIST_IN (area) =
1092 prefix_list_lookup (AFI_IP, PREFIX_NAME_OUT (area));
1093 abr_inv++;
1094 }
1095 }
hassodd669bb2004-05-10 07:43:59 +00001096
1097 /* Schedule ABR task. */
1098 if (IS_OSPF_ABR (ospf) && abr_inv)
1099 ospf_schedule_abr_task (ospf);
1100}
paulcf795c52003-06-19 02:13:25 +00001101
paul4dadc292005-05-06 21:37:42 +00001102static struct ospf_distance *
1103ospf_distance_new (void)
paul718e3742002-12-13 20:15:29 +00001104{
1105 struct ospf_distance *new;
1106 new = XMALLOC (MTYPE_OSPF_DISTANCE, sizeof (struct ospf_distance));
1107 memset (new, 0, sizeof (struct ospf_distance));
1108 return new;
1109}
1110
paul4dadc292005-05-06 21:37:42 +00001111static void
paul718e3742002-12-13 20:15:29 +00001112ospf_distance_free (struct ospf_distance *odistance)
1113{
1114 XFREE (MTYPE_OSPF_DISTANCE, odistance);
1115}
1116
1117int
paul6c835672004-10-11 11:00:30 +00001118ospf_distance_set (struct vty *vty, struct ospf *ospf,
1119 const char *distance_str,
1120 const char *ip_str,
1121 const char *access_list_str)
paul718e3742002-12-13 20:15:29 +00001122{
1123 int ret;
1124 struct prefix_ipv4 p;
1125 u_char distance;
1126 struct route_node *rn;
1127 struct ospf_distance *odistance;
1128
1129 ret = str2prefix_ipv4 (ip_str, &p);
1130 if (ret == 0)
1131 {
1132 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1133 return CMD_WARNING;
1134 }
1135
1136 distance = atoi (distance_str);
1137
1138 /* Get OSPF distance node. */
paul68980082003-03-25 05:07:42 +00001139 rn = route_node_get (ospf->distance_table, (struct prefix *) &p);
paul718e3742002-12-13 20:15:29 +00001140 if (rn->info)
1141 {
1142 odistance = rn->info;
1143 route_unlock_node (rn);
1144 }
1145 else
1146 {
1147 odistance = ospf_distance_new ();
1148 rn->info = odistance;
1149 }
1150
1151 /* Set distance value. */
1152 odistance->distance = distance;
1153
1154 /* Reset access-list configuration. */
1155 if (odistance->access_list)
1156 {
1157 free (odistance->access_list);
1158 odistance->access_list = NULL;
1159 }
1160 if (access_list_str)
1161 odistance->access_list = strdup (access_list_str);
1162
1163 return CMD_SUCCESS;
1164}
1165
1166int
paul6c835672004-10-11 11:00:30 +00001167ospf_distance_unset (struct vty *vty, struct ospf *ospf,
1168 const char *distance_str,
1169 const char *ip_str, char
1170 const *access_list_str)
paul718e3742002-12-13 20:15:29 +00001171{
1172 int ret;
1173 struct prefix_ipv4 p;
1174 u_char distance;
1175 struct route_node *rn;
1176 struct ospf_distance *odistance;
1177
1178 ret = str2prefix_ipv4 (ip_str, &p);
1179 if (ret == 0)
1180 {
1181 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1182 return CMD_WARNING;
1183 }
1184
1185 distance = atoi (distance_str);
1186
paulcf795c52003-06-19 02:13:25 +00001187 rn = route_node_lookup (ospf->distance_table, (struct prefix *) &p);
1188 if (!rn)
paul718e3742002-12-13 20:15:29 +00001189 {
1190 vty_out (vty, "Can't find specified prefix%s", VTY_NEWLINE);
1191 return CMD_WARNING;
1192 }
1193
1194 odistance = rn->info;
1195
1196 if (odistance->access_list)
1197 free (odistance->access_list);
1198 ospf_distance_free (odistance);
1199
1200 rn->info = NULL;
1201 route_unlock_node (rn);
1202 route_unlock_node (rn);
1203
1204 return CMD_SUCCESS;
1205}
1206
1207void
paul68980082003-03-25 05:07:42 +00001208ospf_distance_reset (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +00001209{
1210 struct route_node *rn;
1211 struct ospf_distance *odistance;
1212
paul68980082003-03-25 05:07:42 +00001213 for (rn = route_top (ospf->distance_table); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +00001214 if ((odistance = rn->info) != NULL)
1215 {
paulcf795c52003-06-19 02:13:25 +00001216 if (odistance->access_list)
1217 free (odistance->access_list);
1218 ospf_distance_free (odistance);
1219 rn->info = NULL;
1220 route_unlock_node (rn);
paul718e3742002-12-13 20:15:29 +00001221 }
1222}
1223
1224u_char
1225ospf_distance_apply (struct prefix_ipv4 *p, struct ospf_route *or)
1226{
paul020709f2003-04-04 02:44:16 +00001227 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +00001228
paul020709f2003-04-04 02:44:16 +00001229 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +00001230 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +00001231 return 0;
1232
paul68980082003-03-25 05:07:42 +00001233 if (ospf->distance_intra)
paul718e3742002-12-13 20:15:29 +00001234 if (or->path_type == OSPF_PATH_INTRA_AREA)
paul68980082003-03-25 05:07:42 +00001235 return ospf->distance_intra;
paul718e3742002-12-13 20:15:29 +00001236
paul68980082003-03-25 05:07:42 +00001237 if (ospf->distance_inter)
paul718e3742002-12-13 20:15:29 +00001238 if (or->path_type == OSPF_PATH_INTER_AREA)
paul68980082003-03-25 05:07:42 +00001239 return ospf->distance_inter;
paul718e3742002-12-13 20:15:29 +00001240
paul68980082003-03-25 05:07:42 +00001241 if (ospf->distance_external)
paul718e3742002-12-13 20:15:29 +00001242 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL
paulcf795c52003-06-19 02:13:25 +00001243 || or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
paul68980082003-03-25 05:07:42 +00001244 return ospf->distance_external;
paulcf795c52003-06-19 02:13:25 +00001245
paul68980082003-03-25 05:07:42 +00001246 if (ospf->distance_all)
1247 return ospf->distance_all;
paul718e3742002-12-13 20:15:29 +00001248
1249 return 0;
1250}
1251
1252void
1253ospf_zebra_init ()
1254{
1255 /* Allocate zebra structure. */
1256 zclient = zclient_new ();
1257 zclient_init (zclient, ZEBRA_ROUTE_OSPF);
hasso18a6dce2004-10-03 18:18:34 +00001258 zclient->router_id_update = ospf_router_id_update_zebra;
paul718e3742002-12-13 20:15:29 +00001259 zclient->interface_add = ospf_interface_add;
1260 zclient->interface_delete = ospf_interface_delete;
1261 zclient->interface_up = ospf_interface_state_up;
1262 zclient->interface_down = ospf_interface_state_down;
1263 zclient->interface_address_add = ospf_interface_address_add;
1264 zclient->interface_address_delete = ospf_interface_address_delete;
1265 zclient->ipv4_route_add = ospf_zebra_read_ipv4;
1266 zclient->ipv4_route_delete = ospf_zebra_read_ipv4;
1267
1268 access_list_add_hook (ospf_filter_update);
1269 access_list_delete_hook (ospf_filter_update);
hassodd669bb2004-05-10 07:43:59 +00001270 prefix_list_add_hook (ospf_prefix_list_update);
1271 prefix_list_delete_hook (ospf_prefix_list_update);
paul718e3742002-12-13 20:15:29 +00001272}