blob: 8468aed9eac9ff3e1c2ed4be3c4239aa6f6fefde [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. */
60int
61ospf_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 ();
71 if (ospf != NULL)
72 {
73 if (ospf->t_router_id_update == NULL)
74 OSPF_TIMER_ON (ospf->t_router_id_update, ospf_router_id_update_timer,
75 OSPF_ROUTER_ID_UPDATE_DELAY);
76 }
77 return 0;
78}
paul718e3742002-12-13 20:15:29 +000079
80/* Inteface addition message from zebra. */
81int
82ospf_interface_add (int command, struct zclient *zclient, zebra_size_t length)
83{
84 struct interface *ifp;
paul020709f2003-04-04 02:44:16 +000085 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +000086
87 ifp = zebra_interface_add_read (zclient->ibuf);
88
89 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +000090 zlog_debug ("Zebra: interface add %s index %d flags %ld metric %d mtu %d",
paulcf795c52003-06-19 02:13:25 +000091 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +000092
paulcf795c52003-06-19 02:13:25 +000093 assert (ifp->info);
paulf2c80652002-12-13 21:44:27 +000094
paul718e3742002-12-13 20:15:29 +000095 if (!OSPF_IF_PARAM_CONFIGURED (IF_DEF_PARAMS (ifp), type))
96 {
97 SET_IF_PARAM (IF_DEF_PARAMS (ifp), type);
ajsbc18d612004-12-15 15:07:19 +000098 IF_DEF_PARAMS (ifp)->type = ospf_default_iftype(ifp);
paul718e3742002-12-13 20:15:29 +000099 }
100
paul020709f2003-04-04 02:44:16 +0000101 ospf = ospf_lookup ();
102 if (ospf != NULL)
103 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000104
105#ifdef HAVE_SNMP
106 ospf_snmp_if_update (ifp);
107#endif /* HAVE_SNMP */
108
109 return 0;
110}
111
112int
113ospf_interface_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000114 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000115{
116 struct interface *ifp;
117 struct stream *s;
118 struct route_node *rn;
119
paulcf795c52003-06-19 02:13:25 +0000120 s = zclient->ibuf;
paul718e3742002-12-13 20:15:29 +0000121 /* zebra_interface_state_read() updates interface structure in iflist */
122 ifp = zebra_interface_state_read (s);
123
124 if (ifp == NULL)
125 return 0;
126
127 if (if_is_up (ifp))
128 zlog_warn ("Zebra: got delete of %s, but interface is still up",
paulcf795c52003-06-19 02:13:25 +0000129 ifp->name);
130
paul718e3742002-12-13 20:15:29 +0000131 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000132 zlog_debug
paulcf795c52003-06-19 02:13:25 +0000133 ("Zebra: interface delete %s index %d flags %ld metric %d mtu %d",
134 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +0000135
136#ifdef HAVE_SNMP
137 ospf_snmp_if_delete (ifp);
138#endif /* HAVE_SNMP */
139
140 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
141 if (rn->info)
142 ospf_if_free ((struct ospf_interface *) rn->info);
143
paul718e3742002-12-13 20:15:29 +0000144 return 0;
145}
146
147struct interface *
148zebra_interface_if_lookup (struct stream *s)
149{
150 struct interface *ifp;
151 u_char ifname_tmp[INTERFACE_NAMSIZ];
152
153 /* Read interface name. */
154 stream_get (ifname_tmp, s, INTERFACE_NAMSIZ);
155
156 /* Lookup this by interface index. */
hassoc9e52be2004-09-26 16:09:34 +0000157 ifp = if_lookup_by_name ((char *) ifname_tmp);
paul718e3742002-12-13 20:15:29 +0000158
159 /* If such interface does not exist, indicate an error */
160 if (!ifp)
161 return NULL;
162
163 return ifp;
164}
165
paul718e3742002-12-13 20:15:29 +0000166int
167ospf_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;
171 struct interface if_tmp;
172 struct ospf_interface *oi;
173 struct route_node *rn;
paulcf795c52003-06-19 02:13:25 +0000174
paul718e3742002-12-13 20:15:29 +0000175 ifp = zebra_interface_if_lookup (zclient->ibuf);
176
177 if (ifp == NULL)
178 return 0;
179
180 /* Interface is already up. */
paul2e3b2e42002-12-13 21:03:13 +0000181 if (if_is_operative (ifp))
paul718e3742002-12-13 20:15:29 +0000182 {
183 /* Temporarily keep ifp values. */
184 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 }
paul718e3742002-12-13 20:15:29 +0000199 return 0;
200 }
paulcf795c52003-06-19 02:13:25 +0000201
paul718e3742002-12-13 20:15:29 +0000202 zebra_interface_if_set_value (zclient->ibuf, ifp);
paulcf795c52003-06-19 02:13:25 +0000203
paul718e3742002-12-13 20:15:29 +0000204 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000205 zlog_debug ("Zebra: Interface[%s] state change to up.", ifp->name);
paulcf795c52003-06-19 02:13:25 +0000206
207 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +0000208 {
paulcf795c52003-06-19 02:13:25 +0000209 if ((oi = rn->info) == NULL)
210 continue;
211
paul718e3742002-12-13 20:15:29 +0000212 ospf_if_up (oi);
213 }
paulcf795c52003-06-19 02:13:25 +0000214
paul718e3742002-12-13 20:15:29 +0000215 return 0;
216}
217
218int
219ospf_interface_state_down (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000220 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000221{
222 struct interface *ifp;
223 struct ospf_interface *oi;
224 struct route_node *node;
225
226 ifp = zebra_interface_state_read (zclient->ibuf);
227
228 if (ifp == NULL)
229 return 0;
230
231 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
ajs9b0e25c2004-12-08 19:06:51 +0000232 zlog_debug ("Zebra: Interface[%s] state change to down.", ifp->name);
paul718e3742002-12-13 20:15:29 +0000233
paulcf795c52003-06-19 02:13:25 +0000234 for (node = route_top (IF_OIFS (ifp)); node; node = route_next (node))
paul718e3742002-12-13 20:15:29 +0000235 {
paulcf795c52003-06-19 02:13:25 +0000236 if ((oi = node->info) == NULL)
237 continue;
paul718e3742002-12-13 20:15:29 +0000238 ospf_if_down (oi);
239 }
240
241 return 0;
242}
243
244int
245ospf_interface_address_add (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000246 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000247{
paul020709f2003-04-04 02:44:16 +0000248 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000249 struct connected *c;
250
paul0a589352004-05-08 11:48:26 +0000251 c = zebra_interface_address_read (command, zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000252
253 if (c == NULL)
254 return 0;
255
paul020709f2003-04-04 02:44:16 +0000256 ospf = ospf_lookup ();
257 if (ospf != NULL)
258 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000259
260#ifdef HAVE_SNMP
261 ospf_snmp_if_update (c->ifp);
262#endif /* HAVE_SNMP */
263
264 return 0;
265}
266
267int
268ospf_interface_address_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000269 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000270{
paul020709f2003-04-04 02:44:16 +0000271 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000272 struct connected *c;
273 struct interface *ifp;
274 struct ospf_interface *oi;
275 struct route_node *rn;
276 struct prefix p;
277
paul0a589352004-05-08 11:48:26 +0000278 c = zebra_interface_address_read (command, zclient->ibuf);
paul718e3742002-12-13 20:15:29 +0000279
280 if (c == NULL)
281 return 0;
282
283 ifp = c->ifp;
284 p = *c->address;
285 p.prefixlen = IPV4_MAX_PREFIXLEN;
286
287 rn = route_node_lookup (IF_OIFS (ifp), &p);
paulcf795c52003-06-19 02:13:25 +0000288 if (!rn)
paul718e3742002-12-13 20:15:29 +0000289 return 0;
290
291 assert (rn->info);
292 oi = rn->info;
paulcf795c52003-06-19 02:13:25 +0000293
paul718e3742002-12-13 20:15:29 +0000294 /* Call interface hook functions to clean up */
295 ospf_if_free (oi);
paulcf795c52003-06-19 02:13:25 +0000296
paul718e3742002-12-13 20:15:29 +0000297#ifdef HAVE_SNMP
298 ospf_snmp_if_update (c->ifp);
299#endif /* HAVE_SNMP */
300
301 connected_free (c);
302
paul020709f2003-04-04 02:44:16 +0000303 ospf = ospf_lookup ();
304 if (ospf != NULL)
305 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000306
307 return 0;
308}
paul72357f22003-06-19 02:11:23 +0000309
paul718e3742002-12-13 20:15:29 +0000310void
311ospf_zebra_add (struct prefix_ipv4 *p, struct ospf_route *or)
312{
313 u_char message;
314 u_char distance;
315 u_char flags;
316 int psize;
317 struct stream *s;
318 struct ospf_path *path;
hasso52dc7ee2004-09-23 19:18:23 +0000319 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000320
321 if (zclient->redist[ZEBRA_ROUTE_OSPF])
322 {
323 message = 0;
324 flags = 0;
325
326 /* OSPF pass nexthop and metric */
327 SET_FLAG (message, ZAPI_MESSAGE_NEXTHOP);
328 SET_FLAG (message, ZAPI_MESSAGE_METRIC);
329
330 /* Distance value. */
331 distance = ospf_distance_apply (p, or);
332 if (distance)
paul72357f22003-06-19 02:11:23 +0000333 SET_FLAG (message, ZAPI_MESSAGE_DISTANCE);
paul718e3742002-12-13 20:15:29 +0000334
335 /* Make packet. */
336 s = zclient->obuf;
337 stream_reset (s);
338
339 /* Length place holder. */
340 stream_putw (s, 0);
341
342 /* Put command, type, flags, message. */
343 stream_putc (s, ZEBRA_IPV4_ROUTE_ADD);
344 stream_putc (s, ZEBRA_ROUTE_OSPF);
345 stream_putc (s, flags);
346 stream_putc (s, message);
paulcf795c52003-06-19 02:13:25 +0000347
paul718e3742002-12-13 20:15:29 +0000348 /* Put prefix information. */
349 psize = PSIZE (p->prefixlen);
350 stream_putc (s, p->prefixlen);
paulcf795c52003-06-19 02:13:25 +0000351 stream_write (s, (u_char *) & p->prefix, psize);
paul718e3742002-12-13 20:15:29 +0000352
353 /* Nexthop count. */
paul96735ee2003-08-10 02:51:22 +0000354 stream_putc (s, or->paths->count);
paul718e3742002-12-13 20:15:29 +0000355
356 /* Nexthop, ifindex, distance and metric information. */
paul96735ee2003-08-10 02:51:22 +0000357 for (node = listhead (or->paths); node; nextnode (node))
paul72357f22003-06-19 02:11:23 +0000358 {
paulcf795c52003-06-19 02:13:25 +0000359 path = getdata (node);
paul718e3742002-12-13 20:15:29 +0000360
paulcf795c52003-06-19 02:13:25 +0000361 if (path->nexthop.s_addr != INADDR_ANY)
362 {
363 stream_putc (s, ZEBRA_NEXTHOP_IPV4);
364 stream_put_in_addr (s, &path->nexthop);
365 }
366 else
367 {
368 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
369 if (path->oi)
370 stream_putl (s, path->oi->ifp->ifindex);
371 else
372 stream_putl (s, 0);
373 }
paul72357f22003-06-19 02:11:23 +0000374
375 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
376 {
ajs9b0e25c2004-12-08 19:06:51 +0000377 zlog_debug ("Zebra: Route add %s/%d nexthop %s",
paulcf795c52003-06-19 02:13:25 +0000378 inet_ntoa (p->prefix),
379 p->prefixlen, inet_ntoa (path->nexthop));
380 }
381 }
paul718e3742002-12-13 20:15:29 +0000382
383 if (CHECK_FLAG (message, ZAPI_MESSAGE_DISTANCE))
paulcf795c52003-06-19 02:13:25 +0000384 stream_putc (s, distance);
paul718e3742002-12-13 20:15:29 +0000385 if (CHECK_FLAG (message, ZAPI_MESSAGE_METRIC))
paulcf795c52003-06-19 02:13:25 +0000386 {
387 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL)
388 stream_putl (s, or->cost + or->u.ext.type2_cost);
389 else if (or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
390 stream_putl (s, or->u.ext.type2_cost);
391 else
392 stream_putl (s, or->cost);
393 }
paul718e3742002-12-13 20:15:29 +0000394
395 stream_putw_at (s, 0, stream_get_endp (s));
396
397 writen (zclient->sock, s->data, stream_get_endp (s));
paul718e3742002-12-13 20:15:29 +0000398 }
399}
400
401void
402ospf_zebra_delete (struct prefix_ipv4 *p, struct ospf_route *or)
403{
404 struct zapi_ipv4 api;
paul72357f22003-06-19 02:11:23 +0000405 struct ospf_path *path;
406 struct in_addr *nexthop;
hasso52dc7ee2004-09-23 19:18:23 +0000407 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000408
409 if (zclient->redist[ZEBRA_ROUTE_OSPF])
410 {
411 api.type = ZEBRA_ROUTE_OSPF;
412 api.flags = 0;
413 api.message = 0;
paul72357f22003-06-19 02:11:23 +0000414 api.ifindex_num = 0;
415 api.nexthop_num = 0;
paul718e3742002-12-13 20:15:29 +0000416
paul96735ee2003-08-10 02:51:22 +0000417 for (node = listhead (or->paths); node; nextnode (node))
paulcf795c52003-06-19 02:13:25 +0000418 {
419 path = getdata (node);
paul718e3742002-12-13 20:15:29 +0000420
paulcf795c52003-06-19 02:13:25 +0000421 if (path->nexthop.s_addr != INADDR_ANY)
422 {
pauld8e1d6b2003-08-10 04:04:41 +0000423 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
paulcf795c52003-06-19 02:13:25 +0000424 api.nexthop_num = 1;
425 nexthop = &path->nexthop;
426 api.nexthop = &nexthop;
427 }
hasso2db3d052004-02-11 21:52:13 +0000428 else if (ospf_if_exists(path->oi) && (path->oi->ifp))
paulbb8ff1e2003-08-12 06:00:30 +0000429 {
430 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
431 api.ifindex_num = 1;
432 api.ifindex = &path->oi->ifp->ifindex;
433 }
434 else if ( IS_DEBUG_OSPF(zebra,ZEBRA_REDISTRIBUTE) )
435 {
ajs9b0e25c2004-12-08 19:06:51 +0000436 zlog_debug("Zebra: no ifp %s %d",
paulbb8ff1e2003-08-12 06:00:30 +0000437 inet_ntoa(p->prefix),
438 p->prefixlen);
439 }
paul72357f22003-06-19 02:11:23 +0000440
paul0a589352004-05-08 11:48:26 +0000441 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
paul72357f22003-06-19 02:11:23 +0000442
443 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.nexthop_num)
paulcf795c52003-06-19 02:13:25 +0000444 {
ajs9b0e25c2004-12-08 19:06:51 +0000445 zlog_debug ("Zebra: Route delete %s/%d nexthop %s",
paulcf795c52003-06-19 02:13:25 +0000446 inet_ntoa (p->prefix),
447 p->prefixlen, inet_ntoa (**api.nexthop));
448 }
paulbb8ff1e2003-08-12 06:00:30 +0000449 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.ifindex_num)
450 {
ajs9b0e25c2004-12-08 19:06:51 +0000451 zlog_debug ("Zebra: Route delete %s/%d ifindex %d",
paulbb8ff1e2003-08-12 06:00:30 +0000452 inet_ntoa (p->prefix),
453 p->prefixlen, *api.ifindex);
454 }
paulcf795c52003-06-19 02:13:25 +0000455 }
paul718e3742002-12-13 20:15:29 +0000456 }
457}
458
459void
460ospf_zebra_add_discard (struct prefix_ipv4 *p)
461{
462 struct zapi_ipv4 api;
463
464 if (zclient->redist[ZEBRA_ROUTE_OSPF])
465 {
466 api.type = ZEBRA_ROUTE_OSPF;
467 api.flags = ZEBRA_FLAG_BLACKHOLE;
468 api.message = 0;
469 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
470 api.nexthop_num = 0;
471 api.ifindex_num = 0;
472
paul0a589352004-05-08 11:48:26 +0000473 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_ADD, zclient, p, &api);
paul718e3742002-12-13 20:15:29 +0000474 }
475}
476
477void
478ospf_zebra_delete_discard (struct prefix_ipv4 *p)
479{
480 struct zapi_ipv4 api;
481
482 if (zclient->redist[ZEBRA_ROUTE_OSPF])
483 {
484 api.type = ZEBRA_ROUTE_OSPF;
485 api.flags = ZEBRA_FLAG_BLACKHOLE;
486 api.message = 0;
487 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
488 api.nexthop_num = 0;
489 api.ifindex_num = 0;
490
paul0a589352004-05-08 11:48:26 +0000491 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
paul72357f22003-06-19 02:11:23 +0000492
493 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000494 zlog_debug ("Zebra: Route delete discard %s/%d",
paulcf795c52003-06-19 02:13:25 +0000495 inet_ntoa (p->prefix), p->prefixlen);
paul72357f22003-06-19 02:11:23 +0000496
paul718e3742002-12-13 20:15:29 +0000497 }
498}
499
500int
501ospf_is_type_redistributed (int type)
502{
503 return (DEFAULT_ROUTE_TYPE (type)) ?
504 zclient->default_information : zclient->redist[type];
505}
506
507int
paul020709f2003-04-04 02:44:16 +0000508ospf_redistribute_set (struct ospf *ospf, int type, int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000509{
510 int force = 0;
paulcf795c52003-06-19 02:13:25 +0000511
paul718e3742002-12-13 20:15:29 +0000512 if (ospf_is_type_redistributed (type))
513 {
paul68980082003-03-25 05:07:42 +0000514 if (mtype != ospf->dmetric[type].type)
paulcf795c52003-06-19 02:13:25 +0000515 {
516 ospf->dmetric[type].type = mtype;
517 force = LSA_REFRESH_FORCE;
518 }
paul68980082003-03-25 05:07:42 +0000519 if (mvalue != ospf->dmetric[type].value)
paulcf795c52003-06-19 02:13:25 +0000520 {
521 ospf->dmetric[type].value = mvalue;
522 force = LSA_REFRESH_FORCE;
523 }
524
paul68980082003-03-25 05:07:42 +0000525 ospf_external_lsa_refresh_type (ospf, type, force);
paulcf795c52003-06-19 02:13:25 +0000526
paul718e3742002-12-13 20:15:29 +0000527 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000528 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000529 LOOKUP (ospf_redistributed_proto, type),
530 metric_type (ospf, type), metric_value (ospf, type));
531
paul718e3742002-12-13 20:15:29 +0000532 return CMD_SUCCESS;
533 }
534
paul68980082003-03-25 05:07:42 +0000535 ospf->dmetric[type].type = mtype;
536 ospf->dmetric[type].value = mvalue;
paul718e3742002-12-13 20:15:29 +0000537
paul0a589352004-05-08 11:48:26 +0000538 zclient_redistribute (ZEBRA_REDISTRIBUTE_ADD, zclient, type);
paul718e3742002-12-13 20:15:29 +0000539
540 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000541 zlog_debug ("Redistribute[%s]: Start Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000542 LOOKUP (ospf_redistributed_proto, type),
543 metric_type (ospf, type), metric_value (ospf, type));
544
paul68980082003-03-25 05:07:42 +0000545 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000546
547 return CMD_SUCCESS;
548}
549
550int
paul020709f2003-04-04 02:44:16 +0000551ospf_redistribute_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000552{
553 if (type == zclient->redist_default)
554 return CMD_SUCCESS;
555
paulcf795c52003-06-19 02:13:25 +0000556 if (!ospf_is_type_redistributed (type))
paul718e3742002-12-13 20:15:29 +0000557 return CMD_SUCCESS;
558
paul0a589352004-05-08 11:48:26 +0000559 zclient_redistribute (ZEBRA_REDISTRIBUTE_DELETE, zclient, type);
paulcf795c52003-06-19 02:13:25 +0000560
paul718e3742002-12-13 20:15:29 +0000561 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000562 zlog_debug ("Redistribute[%s]: Stop",
paulcf795c52003-06-19 02:13:25 +0000563 LOOKUP (ospf_redistributed_proto, type));
paul718e3742002-12-13 20:15:29 +0000564
paul68980082003-03-25 05:07:42 +0000565 ospf->dmetric[type].type = -1;
566 ospf->dmetric[type].value = -1;
paul718e3742002-12-13 20:15:29 +0000567
568 /* Remove the routes from OSPF table. */
569 ospf_redistribute_withdraw (type);
570
paul68980082003-03-25 05:07:42 +0000571 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000572
573 return CMD_SUCCESS;
574}
575
576int
paul020709f2003-04-04 02:44:16 +0000577ospf_redistribute_default_set (struct ospf *ospf, int originate,
paulcf795c52003-06-19 02:13:25 +0000578 int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000579{
580 int force = 0;
paul020709f2003-04-04 02:44:16 +0000581
paul718e3742002-12-13 20:15:29 +0000582 if (ospf_is_type_redistributed (DEFAULT_ROUTE))
583 {
paul68980082003-03-25 05:07:42 +0000584 if (mtype != ospf->dmetric[DEFAULT_ROUTE].type)
paulcf795c52003-06-19 02:13:25 +0000585 {
586 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
587 force = 1;
588 }
paul68980082003-03-25 05:07:42 +0000589 if (mvalue != ospf->dmetric[DEFAULT_ROUTE].value)
paulcf795c52003-06-19 02:13:25 +0000590 {
591 force = 1;
592 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
593 }
594
paul68980082003-03-25 05:07:42 +0000595 ospf_external_lsa_refresh_default (ospf);
paulcf795c52003-06-19 02:13:25 +0000596
paul718e3742002-12-13 20:15:29 +0000597 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000598 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000599 LOOKUP (ospf_redistributed_proto, DEFAULT_ROUTE),
600 metric_type (ospf, DEFAULT_ROUTE),
601 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000602 return CMD_SUCCESS;
603 }
604
paul68980082003-03-25 05:07:42 +0000605 ospf->default_originate = originate;
606 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
607 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
paul718e3742002-12-13 20:15:29 +0000608
paul0a589352004-05-08 11:48:26 +0000609 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_ADD, zclient);
paulcf795c52003-06-19 02:13:25 +0000610
paul718e3742002-12-13 20:15:29 +0000611 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000612 zlog_debug ("Redistribute[DEFAULT]: Start Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000613 metric_type (ospf, DEFAULT_ROUTE),
614 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000615
paul68980082003-03-25 05:07:42 +0000616 if (ospf->router_id.s_addr == 0)
617 ospf->external_origin |= (1 << DEFAULT_ROUTE);
paul718e3742002-12-13 20:15:29 +0000618 else
619 thread_add_timer (master, ospf_default_originate_timer,
paulcf795c52003-06-19 02:13:25 +0000620 &ospf->default_originate, 1);
paul718e3742002-12-13 20:15:29 +0000621
paul68980082003-03-25 05:07:42 +0000622 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000623
624 return CMD_SUCCESS;
625}
626
627int
paul020709f2003-04-04 02:44:16 +0000628ospf_redistribute_default_unset (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +0000629{
630 if (!ospf_is_type_redistributed (DEFAULT_ROUTE))
631 return CMD_SUCCESS;
632
paul68980082003-03-25 05:07:42 +0000633 ospf->default_originate = DEFAULT_ORIGINATE_NONE;
634 ospf->dmetric[DEFAULT_ROUTE].type = -1;
635 ospf->dmetric[DEFAULT_ROUTE].value = -1;
paul718e3742002-12-13 20:15:29 +0000636
paul0a589352004-05-08 11:48:26 +0000637 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_DELETE, zclient);
paul718e3742002-12-13 20:15:29 +0000638
639 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000640 zlog_debug ("Redistribute[DEFAULT]: Stop");
paulcf795c52003-06-19 02:13:25 +0000641
paul68980082003-03-25 05:07:42 +0000642 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000643
644 return CMD_SUCCESS;
645}
646
647int
paul020709f2003-04-04 02:44:16 +0000648ospf_external_lsa_originate_check (struct ospf *ospf,
paulcf795c52003-06-19 02:13:25 +0000649 struct external_info *ei)
paul718e3742002-12-13 20:15:29 +0000650{
651 /* If prefix is multicast, then do not originate LSA. */
652 if (IN_MULTICAST (htonl (ei->p.prefix.s_addr)))
653 {
654 zlog_info ("LSA[Type5:%s]: Not originate AS-external-LSA, "
paulcf795c52003-06-19 02:13:25 +0000655 "Prefix belongs multicast", inet_ntoa (ei->p.prefix));
paul718e3742002-12-13 20:15:29 +0000656 return 0;
657 }
658
659 /* Take care of default-originate. */
660 if (is_prefix_default (&ei->p))
paul68980082003-03-25 05:07:42 +0000661 if (ospf->default_originate == DEFAULT_ORIGINATE_NONE)
paul718e3742002-12-13 20:15:29 +0000662 {
paulcf795c52003-06-19 02:13:25 +0000663 zlog_info ("LSA[Type5:0.0.0.0]: Not originate AS-exntenal-LSA "
664 "for default");
665 return 0;
paul718e3742002-12-13 20:15:29 +0000666 }
667
668 return 1;
669}
670
671/* If connected prefix is OSPF enable interface, then do not announce. */
672int
paulcf795c52003-06-19 02:13:25 +0000673ospf_distribute_check_connected (struct ospf *ospf, struct external_info *ei)
paul718e3742002-12-13 20:15:29 +0000674{
675 struct route_node *rn;
676
paul68980082003-03-25 05:07:42 +0000677 for (rn = route_top (ospf->networks); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +0000678 if (rn->info != NULL)
paulcf795c52003-06-19 02:13:25 +0000679 if (prefix_match (&rn->p, (struct prefix *) &ei->p))
680 {
681 route_unlock_node (rn);
682 return 0;
683 }
paul718e3742002-12-13 20:15:29 +0000684
685 return 1;
686}
687
688/* return 1 if external LSA must be originated, 0 otherwise */
689int
paul68980082003-03-25 05:07:42 +0000690ospf_redistribute_check (struct ospf *ospf,
paulcf795c52003-06-19 02:13:25 +0000691 struct external_info *ei, int *changed)
paul718e3742002-12-13 20:15:29 +0000692{
693 struct route_map_set_values save_values;
694 struct prefix_ipv4 *p = &ei->p;
695 u_char type = is_prefix_default (&ei->p) ? DEFAULT_ROUTE : ei->type;
paulcf795c52003-06-19 02:13:25 +0000696
paul718e3742002-12-13 20:15:29 +0000697 if (changed)
698 *changed = 0;
699
paul020709f2003-04-04 02:44:16 +0000700 if (!ospf_external_lsa_originate_check (ospf, ei))
paul718e3742002-12-13 20:15:29 +0000701 return 0;
702
703 /* Take care connected route. */
paul68980082003-03-25 05:07:42 +0000704 if (type == ZEBRA_ROUTE_CONNECT &&
705 !ospf_distribute_check_connected (ospf, ei))
paul718e3742002-12-13 20:15:29 +0000706 return 0;
707
paul020709f2003-04-04 02:44:16 +0000708 if (!DEFAULT_ROUTE_TYPE (type) && DISTRIBUTE_NAME (ospf, type))
paul718e3742002-12-13 20:15:29 +0000709 /* distirbute-list exists, but access-list may not? */
paul020709f2003-04-04 02:44:16 +0000710 if (DISTRIBUTE_LIST (ospf, type))
711 if (access_list_apply (DISTRIBUTE_LIST (ospf, type), p) == FILTER_DENY)
paulcf795c52003-06-19 02:13:25 +0000712 {
713 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000714 zlog_debug ("Redistribute[%s]: %s/%d filtered by ditribute-list.",
paulcf795c52003-06-19 02:13:25 +0000715 LOOKUP (ospf_redistributed_proto, type),
716 inet_ntoa (p->prefix), p->prefixlen);
717 return 0;
718 }
paul718e3742002-12-13 20:15:29 +0000719
720 save_values = ei->route_map_set;
721 ospf_reset_route_map_set_values (&ei->route_map_set);
paulcf795c52003-06-19 02:13:25 +0000722
paul718e3742002-12-13 20:15:29 +0000723 /* apply route-map if needed */
paul020709f2003-04-04 02:44:16 +0000724 if (ROUTEMAP_NAME (ospf, type))
paul718e3742002-12-13 20:15:29 +0000725 {
726 int ret;
727
paulcf795c52003-06-19 02:13:25 +0000728 ret = route_map_apply (ROUTEMAP (ospf, type), (struct prefix *) p,
729 RMAP_OSPF, ei);
paul718e3742002-12-13 20:15:29 +0000730
731 if (ret == RMAP_DENYMATCH)
paulcf795c52003-06-19 02:13:25 +0000732 {
733 ei->route_map_set = save_values;
734 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
ajs9b0e25c2004-12-08 19:06:51 +0000735 zlog_debug ("Redistribute[%s]: %s/%d filtered by route-map.",
paulcf795c52003-06-19 02:13:25 +0000736 LOOKUP (ospf_redistributed_proto, type),
737 inet_ntoa (p->prefix), p->prefixlen);
738 return 0;
739 }
740
paul718e3742002-12-13 20:15:29 +0000741 /* check if 'route-map set' changed something */
742 if (changed)
paulcf795c52003-06-19 02:13:25 +0000743 *changed = !ospf_route_map_set_compare (&ei->route_map_set,
744 &save_values);
paul718e3742002-12-13 20:15:29 +0000745 }
746
747 return 1;
748}
749
750/* OSPF route-map set for redistribution */
751void
paul6c835672004-10-11 11:00:30 +0000752ospf_routemap_set (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000753{
paul020709f2003-04-04 02:44:16 +0000754 if (ROUTEMAP_NAME (ospf, type))
755 free (ROUTEMAP_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000756
paul020709f2003-04-04 02:44:16 +0000757 ROUTEMAP_NAME (ospf, type) = strdup (name);
758 ROUTEMAP (ospf, type) = route_map_lookup_by_name (name);
paul718e3742002-12-13 20:15:29 +0000759}
760
761void
paul020709f2003-04-04 02:44:16 +0000762ospf_routemap_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000763{
paul020709f2003-04-04 02:44:16 +0000764 if (ROUTEMAP_NAME (ospf, type))
765 free (ROUTEMAP_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000766
paul020709f2003-04-04 02:44:16 +0000767 ROUTEMAP_NAME (ospf, type) = NULL;
768 ROUTEMAP (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000769}
770
771/* Zebra route add and delete treatment. */
772int
773ospf_zebra_read_ipv4 (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000774 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000775{
776 struct stream *s;
777 struct zapi_ipv4 api;
778 unsigned long ifindex;
779 struct in_addr nexthop;
780 struct prefix_ipv4 p;
781 struct external_info *ei;
paul020709f2003-04-04 02:44:16 +0000782 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000783
784 s = zclient->ibuf;
785 ifindex = 0;
786 nexthop.s_addr = 0;
787
788 /* Type, flags, message. */
789 api.type = stream_getc (s);
790 api.flags = stream_getc (s);
791 api.message = stream_getc (s);
792
793 /* IPv4 prefix. */
794 memset (&p, 0, sizeof (struct prefix_ipv4));
795 p.family = AF_INET;
796 p.prefixlen = stream_getc (s);
797 stream_get (&p.prefix, s, PSIZE (p.prefixlen));
798
hasso8585d4e2004-04-20 17:25:12 +0000799 if (IPV4_NET127(ntohl(p.prefix.s_addr)))
800 return 0;
801
paul718e3742002-12-13 20:15:29 +0000802 /* Nexthop, ifindex, distance, metric. */
803 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP))
804 {
805 api.nexthop_num = stream_getc (s);
806 nexthop.s_addr = stream_get_ipv4 (s);
807 }
808 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_IFINDEX))
809 {
810 api.ifindex_num = stream_getc (s);
gdt6a8da852004-02-13 17:40:51 +0000811 /* XXX assert(api.ifindex_num == 1); */
paul718e3742002-12-13 20:15:29 +0000812 ifindex = stream_getl (s);
813 }
814 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_DISTANCE))
815 api.distance = stream_getc (s);
816 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_METRIC))
817 api.metric = stream_getl (s);
818
paul020709f2003-04-04 02:44:16 +0000819 ospf = ospf_lookup ();
820 if (ospf == NULL)
821 return 0;
822
paul718e3742002-12-13 20:15:29 +0000823 if (command == ZEBRA_IPV4_ROUTE_ADD)
824 {
paul7021c422003-07-15 12:52:22 +0000825 /* XXX|HACK|TODO|FIXME:
hassoa0a39762004-04-23 08:51:10 +0000826 * Maybe we should ignore reject/blackhole routes? Testing shows that
827 * there is no problems though and this is only way to "summarize"
828 * routes in ASBR at the moment. Maybe we need just a better generalised
829 * solution for these types?
830 *
831 * if ( CHECK_FLAG (api.flags, ZEBRA_FLAG_BLACKHOLE)
832 * || CHECK_FLAG (api.flags, ZEBRA_FLAG_REJECT))
833 * return 0;
paul7021c422003-07-15 12:52:22 +0000834 */
paul7021c422003-07-15 12:52:22 +0000835
paul718e3742002-12-13 20:15:29 +0000836 ei = ospf_external_info_add (api.type, p, ifindex, nexthop);
837
paul68980082003-03-25 05:07:42 +0000838 if (ospf->router_id.s_addr == 0)
paulcf795c52003-06-19 02:13:25 +0000839 /* Set flags to generate AS-external-LSA originate event
840 for each redistributed protocols later. */
841 ospf->external_origin |= (1 << api.type);
paul718e3742002-12-13 20:15:29 +0000842 else
paulcf795c52003-06-19 02:13:25 +0000843 {
844 if (ei)
845 {
846 if (is_prefix_default (&p))
847 ospf_external_lsa_refresh_default (ospf);
848 else
849 {
850 struct ospf_lsa *current;
paul718e3742002-12-13 20:15:29 +0000851
paulcf795c52003-06-19 02:13:25 +0000852 current = ospf_external_info_find_lsa (ospf, &ei->p);
853 if (!current)
854 ospf_external_lsa_originate (ospf, ei);
855 else if (IS_LSA_MAXAGE (current))
856 ospf_external_lsa_refresh (ospf, current,
857 ei, LSA_REFRESH_FORCE);
858 else
859 zlog_warn ("ospf_zebra_read_ipv4() : %s already exists",
860 inet_ntoa (p.prefix));
861 }
862 }
863 }
paul718e3742002-12-13 20:15:29 +0000864 }
paulcf795c52003-06-19 02:13:25 +0000865 else /* if (command == ZEBRA_IPV4_ROUTE_DELETE) */
paul718e3742002-12-13 20:15:29 +0000866 {
867 ospf_external_info_delete (api.type, p);
paul7021c422003-07-15 12:52:22 +0000868 if (is_prefix_default (&p))
paulcf795c52003-06-19 02:13:25 +0000869 ospf_external_lsa_refresh_default (ospf);
paul7021c422003-07-15 12:52:22 +0000870 else
871 ospf_external_lsa_flush (ospf, api.type, &p, ifindex, nexthop);
paul718e3742002-12-13 20:15:29 +0000872 }
873
874 return 0;
875}
paul718e3742002-12-13 20:15:29 +0000876
paulcf795c52003-06-19 02:13:25 +0000877
paul718e3742002-12-13 20:15:29 +0000878int
paul6c835672004-10-11 11:00:30 +0000879ospf_distribute_list_out_set (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000880{
881 /* Lookup access-list for distribute-list. */
paul020709f2003-04-04 02:44:16 +0000882 DISTRIBUTE_LIST (ospf, type) = access_list_lookup (AFI_IP, name);
paul718e3742002-12-13 20:15:29 +0000883
884 /* Clear previous distribute-name. */
paul020709f2003-04-04 02:44:16 +0000885 if (DISTRIBUTE_NAME (ospf, type))
886 free (DISTRIBUTE_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000887
888 /* Set distribute-name. */
paul020709f2003-04-04 02:44:16 +0000889 DISTRIBUTE_NAME (ospf, type) = strdup (name);
paul718e3742002-12-13 20:15:29 +0000890
891 /* If access-list have been set, schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000892 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000893 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000894
895 return CMD_SUCCESS;
896}
897
898int
paul6c835672004-10-11 11:00:30 +0000899ospf_distribute_list_out_unset (struct ospf *ospf, int type, const char *name)
paul718e3742002-12-13 20:15:29 +0000900{
901 /* Schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000902 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000903 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000904
905 /* Unset distribute-list. */
paul020709f2003-04-04 02:44:16 +0000906 DISTRIBUTE_LIST (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000907
908 /* Clear distribute-name. */
paul020709f2003-04-04 02:44:16 +0000909 if (DISTRIBUTE_NAME (ospf, type))
910 free (DISTRIBUTE_NAME (ospf, type));
paulcf795c52003-06-19 02:13:25 +0000911
paul020709f2003-04-04 02:44:16 +0000912 DISTRIBUTE_NAME (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000913
914 return CMD_SUCCESS;
915}
916
917/* distribute-list update timer. */
918int
919ospf_distribute_list_update_timer (struct thread *thread)
920{
921 struct route_node *rn;
922 struct external_info *ei;
923 struct route_table *rt;
924 struct ospf_lsa *lsa;
paul64511f32004-10-31 18:01:13 +0000925 int type;
paul020709f2003-04-04 02:44:16 +0000926 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000927
928 type = (int) THREAD_ARG (thread);
paul64511f32004-10-31 18:01:13 +0000929 assert (type < ZEBRA_ROUTE_MAX);
930
paul718e3742002-12-13 20:15:29 +0000931 rt = EXTERNAL_INFO (type);
932
paul020709f2003-04-04 02:44:16 +0000933 ospf = ospf_lookup ();
934 if (ospf == NULL)
935 return 0;
936
paul68980082003-03-25 05:07:42 +0000937 ospf->t_distribute_update = NULL;
paul718e3742002-12-13 20:15:29 +0000938
939 zlog_info ("Zebra[Redistribute]: distribute-list update timer fired!");
940
941 /* foreach all external info. */
942 if (rt)
943 for (rn = route_top (rt); rn; rn = route_next (rn))
944 if ((ei = rn->info) != NULL)
paulcf795c52003-06-19 02:13:25 +0000945 {
946 if (is_prefix_default (&ei->p))
947 ospf_external_lsa_refresh_default (ospf);
948 else if ((lsa = ospf_external_info_find_lsa (ospf, &ei->p)))
949 ospf_external_lsa_refresh (ospf, lsa, ei, LSA_REFRESH_IF_CHANGED);
950 else
951 ospf_external_lsa_originate (ospf, ei);
952 }
paul718e3742002-12-13 20:15:29 +0000953 return 0;
954}
955
956#define OSPF_DISTRIBUTE_UPDATE_DELAY 5
957
958/* Update distribute-list and set timer to apply access-list. */
959void
paul68980082003-03-25 05:07:42 +0000960ospf_distribute_list_update (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000961{
962 struct route_table *rt;
paulcf795c52003-06-19 02:13:25 +0000963
paul718e3742002-12-13 20:15:29 +0000964 /* External info does not exist. */
965 if (!(rt = EXTERNAL_INFO (type)))
966 return;
967
968 /* If exists previously invoked thread, then cancel it. */
paul68980082003-03-25 05:07:42 +0000969 if (ospf->t_distribute_update)
970 OSPF_TIMER_OFF (ospf->t_distribute_update);
paul718e3742002-12-13 20:15:29 +0000971
972 /* Set timer. */
paul68980082003-03-25 05:07:42 +0000973 ospf->t_distribute_update =
paul718e3742002-12-13 20:15:29 +0000974 thread_add_timer (master, ospf_distribute_list_update_timer,
paulcf795c52003-06-19 02:13:25 +0000975 (void *) type, OSPF_DISTRIBUTE_UPDATE_DELAY);
paul718e3742002-12-13 20:15:29 +0000976}
977
978/* If access-list is updated, apply some check. */
979void
980ospf_filter_update (struct access_list *access)
981{
paul020709f2003-04-04 02:44:16 +0000982 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000983 int type;
984 int abr_inv = 0;
985 struct ospf_area *area;
hasso52dc7ee2004-09-23 19:18:23 +0000986 struct listnode *node;
paul718e3742002-12-13 20:15:29 +0000987
988 /* If OSPF instatnce does not exist, return right now. */
paul020709f2003-04-04 02:44:16 +0000989 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +0000990 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +0000991 return;
992
paul718e3742002-12-13 20:15:29 +0000993 /* Update distribute-list, and apply filter. */
994 for (type = 0; type < ZEBRA_ROUTE_MAX; type++)
995 {
paul020709f2003-04-04 02:44:16 +0000996 if (ROUTEMAP (ospf, type) != NULL)
paulcf795c52003-06-19 02:13:25 +0000997 {
998 /* if route-map is not NULL it may be using this access list */
999 ospf_distribute_list_update (ospf, type);
1000 continue;
1001 }
1002
paul718e3742002-12-13 20:15:29 +00001003
paul020709f2003-04-04 02:44:16 +00001004 if (DISTRIBUTE_NAME (ospf, type))
paulcf795c52003-06-19 02:13:25 +00001005 {
1006 /* Keep old access-list for distribute-list. */
1007 struct access_list *old = DISTRIBUTE_LIST (ospf, type);
1008
1009 /* Update access-list for distribute-list. */
1010 DISTRIBUTE_LIST (ospf, type) =
1011 access_list_lookup (AFI_IP, DISTRIBUTE_NAME (ospf, type));
1012
1013 /* No update for this distribute type. */
1014 if (old == NULL && DISTRIBUTE_LIST (ospf, type) == NULL)
1015 continue;
1016
1017 /* Schedule distribute-list update timer. */
1018 if (DISTRIBUTE_LIST (ospf, type) == NULL ||
1019 strcmp (DISTRIBUTE_NAME (ospf, type), access->name) == 0)
1020 ospf_distribute_list_update (ospf, type);
1021 }
paul718e3742002-12-13 20:15:29 +00001022 }
1023
1024 /* Update Area access-list. */
paul68980082003-03-25 05:07:42 +00001025 for (node = listhead (ospf->areas); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +00001026 if ((area = getdata (node)) != NULL)
1027 {
paulcf795c52003-06-19 02:13:25 +00001028 if (EXPORT_NAME (area))
1029 {
1030 EXPORT_LIST (area) = NULL;
1031 abr_inv++;
1032 }
paul718e3742002-12-13 20:15:29 +00001033
paulcf795c52003-06-19 02:13:25 +00001034 if (IMPORT_NAME (area))
1035 {
1036 IMPORT_LIST (area) = NULL;
1037 abr_inv++;
1038 }
paul718e3742002-12-13 20:15:29 +00001039 }
1040
1041 /* Schedule ABR tasks -- this will be changed -- takada. */
paul020709f2003-04-04 02:44:16 +00001042 if (IS_OSPF_ABR (ospf) && abr_inv)
paul68980082003-03-25 05:07:42 +00001043 ospf_schedule_abr_task (ospf);
paul718e3742002-12-13 20:15:29 +00001044}
hassodd669bb2004-05-10 07:43:59 +00001045
1046/* If prefix-list is updated, do some updates. */
1047void
1048ospf_prefix_list_update (struct prefix_list *plist)
1049{
1050 struct ospf *ospf;
1051 int type;
1052 int abr_inv = 0;
1053 struct ospf_area *area;
hasso52dc7ee2004-09-23 19:18:23 +00001054 struct listnode *node;
hassodd669bb2004-05-10 07:43:59 +00001055
1056 /* If OSPF instatnce does not exist, return right now. */
1057 ospf = ospf_lookup ();
1058 if (ospf == NULL)
1059 return;
1060
1061 /* Update all route-maps which are used as redistribution filters.
1062 * They might use prefix-list.
1063 */
1064 for (type = 0; type < ZEBRA_ROUTE_MAX; type++)
1065 {
1066 if (ROUTEMAP (ospf, type) != NULL)
1067 {
1068 /* If route-map is not NULL it may be using this prefix list */
1069 ospf_distribute_list_update (ospf, type);
1070 continue;
1071 }
1072 }
1073
1074 /* Update area filter-lists. */
1075 for (node = listhead (ospf->areas); node; nextnode (node))
1076 if ((area = getdata (node)) != NULL)
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 }
1086
1087 /* 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 }
1096
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
paul718e3742002-12-13 20:15:29 +00001102struct ospf_distance *
1103ospf_distance_new ()
1104{
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
1111void
1112ospf_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}