]> git.proxmox.com Git - mirror_frr.git/blob - ospfd/ospf_zebra.c
ospfd: Make sure ospf_distribute_list_update_timer() eventually runs.
[mirror_frr.git] / ospfd / ospf_zebra.c
1 /*
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"
35 #include "plist.h"
36 #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. */
53 struct zclient *zclient = NULL;
54
55 /* For registering threads. */
56 extern struct thread_master *master;
57 struct in_addr router_id_zebra;
58
59 /* Router-id update message from zebra. */
60 static int
61 ospf_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 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
69 {
70 char buf[128];
71 prefix2str(&router_id, buf, sizeof(buf));
72 zlog_debug("Zebra rcvd: router id update %s", buf);
73 }
74
75 router_id_zebra = router_id.u.prefix4;
76
77 ospf = ospf_lookup ();
78
79 if (ospf != NULL)
80 ospf_router_id_update (ospf);
81
82 return 0;
83 }
84
85 /* Inteface addition message from zebra. */
86 static int
87 ospf_interface_add (int command, struct zclient *zclient, zebra_size_t length)
88 {
89 struct interface *ifp;
90
91 ifp = zebra_interface_add_read (zclient->ibuf);
92
93 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
94 zlog_debug ("Zebra: interface add %s index %d flags %llx metric %d mtu %d",
95 ifp->name, ifp->ifindex, (unsigned long long)ifp->flags,
96 ifp->metric, ifp->mtu);
97
98 assert (ifp->info);
99
100 if (!OSPF_IF_PARAM_CONFIGURED (IF_DEF_PARAMS (ifp), type))
101 {
102 SET_IF_PARAM (IF_DEF_PARAMS (ifp), type);
103 IF_DEF_PARAMS (ifp)->type = ospf_default_iftype(ifp);
104 }
105
106 ospf_if_update (NULL, ifp);
107
108 #ifdef HAVE_SNMP
109 ospf_snmp_if_update (ifp);
110 #endif /* HAVE_SNMP */
111
112 return 0;
113 }
114
115 static int
116 ospf_interface_delete (int command, struct zclient *zclient,
117 zebra_size_t length)
118 {
119 struct interface *ifp;
120 struct stream *s;
121 struct route_node *rn;
122
123 s = zclient->ibuf;
124 /* zebra_interface_state_read() updates interface structure in iflist */
125 ifp = zebra_interface_state_read (s);
126
127 if (ifp == NULL)
128 return 0;
129
130 if (if_is_up (ifp))
131 zlog_warn ("Zebra: got delete of %s, but interface is still up",
132 ifp->name);
133
134 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
135 zlog_debug
136 ("Zebra: interface delete %s index %d flags %lld metric %d mtu %d",
137 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
138
139 #ifdef HAVE_SNMP
140 ospf_snmp_if_delete (ifp);
141 #endif /* HAVE_SNMP */
142
143 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
144 if (rn->info)
145 ospf_if_free ((struct ospf_interface *) rn->info);
146
147 ifp->ifindex = IFINDEX_INTERNAL;
148 return 0;
149 }
150
151 static struct interface *
152 zebra_interface_if_lookup (struct stream *s)
153 {
154 char ifname_tmp[INTERFACE_NAMSIZ];
155
156 /* Read interface name. */
157 stream_get (ifname_tmp, s, INTERFACE_NAMSIZ);
158
159 /* And look it up. */
160 return if_lookup_by_name_len(ifname_tmp,
161 strnlen(ifname_tmp, INTERFACE_NAMSIZ));
162 }
163
164 static int
165 ospf_interface_state_up (int command, struct zclient *zclient,
166 zebra_size_t length)
167 {
168 struct interface *ifp;
169 struct ospf_interface *oi;
170 struct route_node *rn;
171
172 ifp = zebra_interface_if_lookup (zclient->ibuf);
173
174 if (ifp == NULL)
175 return 0;
176
177 /* Interface is already up. */
178 if (if_is_operative (ifp))
179 {
180 /* Temporarily keep ifp values. */
181 struct interface if_tmp;
182 memcpy (&if_tmp, ifp, sizeof (struct interface));
183
184 zebra_interface_if_set_value (zclient->ibuf, ifp);
185
186 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
187 zlog_debug ("Zebra: Interface[%s] state update.", ifp->name);
188
189 if (if_tmp.bandwidth != ifp->bandwidth)
190 {
191 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
192 zlog_debug ("Zebra: Interface[%s] bandwidth change %d -> %d.",
193 ifp->name, if_tmp.bandwidth, ifp->bandwidth);
194
195 ospf_if_recalculate_output_cost (ifp);
196 }
197
198 if (if_tmp.mtu != ifp->mtu)
199 {
200 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
201 zlog_debug ("Zebra: Interface[%s] MTU change %u -> %u.",
202 ifp->name, if_tmp.mtu, ifp->mtu);
203
204 /* Must reset the interface (simulate down/up) when MTU changes. */
205 ospf_if_reset(ifp);
206 }
207 return 0;
208 }
209
210 zebra_interface_if_set_value (zclient->ibuf, ifp);
211
212 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
213 zlog_debug ("Zebra: Interface[%s] state change to up.", ifp->name);
214
215 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
216 {
217 if ((oi = rn->info) == NULL)
218 continue;
219
220 ospf_if_up (oi);
221 }
222
223 return 0;
224 }
225
226 static int
227 ospf_interface_state_down (int command, struct zclient *zclient,
228 zebra_size_t length)
229 {
230 struct interface *ifp;
231 struct ospf_interface *oi;
232 struct route_node *node;
233
234 ifp = zebra_interface_state_read (zclient->ibuf);
235
236 if (ifp == NULL)
237 return 0;
238
239 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
240 zlog_debug ("Zebra: Interface[%s] state change to down.", ifp->name);
241
242 for (node = route_top (IF_OIFS (ifp)); node; node = route_next (node))
243 {
244 if ((oi = node->info) == NULL)
245 continue;
246 ospf_if_down (oi);
247 }
248
249 return 0;
250 }
251
252 static int
253 ospf_interface_address_add (int command, struct zclient *zclient,
254 zebra_size_t length)
255 {
256 struct connected *c;
257
258 c = zebra_interface_address_read (command, zclient->ibuf);
259
260 if (c == NULL)
261 return 0;
262
263 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
264 {
265 char buf[128];
266 prefix2str(c->address, buf, sizeof(buf));
267 zlog_debug("Zebra: interface %s address add %s", c->ifp->name, buf);
268 }
269
270 ospf_if_update (NULL, c->ifp);
271
272 #ifdef HAVE_SNMP
273 ospf_snmp_if_update (c->ifp);
274 #endif /* HAVE_SNMP */
275
276 return 0;
277 }
278
279 static int
280 ospf_interface_address_delete (int command, struct zclient *zclient,
281 zebra_size_t length)
282 {
283 struct connected *c;
284 struct interface *ifp;
285 struct ospf_interface *oi;
286 struct route_node *rn;
287 struct prefix p;
288
289 c = zebra_interface_address_read (command, zclient->ibuf);
290
291 if (c == NULL)
292 return 0;
293
294 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
295 {
296 char buf[128];
297 prefix2str(c->address, buf, sizeof(buf));
298 zlog_debug("Zebra: interface %s address delete %s", c->ifp->name, buf);
299 }
300
301 ifp = c->ifp;
302 p = *c->address;
303 p.prefixlen = IPV4_MAX_PREFIXLEN;
304
305 rn = route_node_lookup (IF_OIFS (ifp), &p);
306 if (!rn)
307 {
308 connected_free (c);
309 return 0;
310 }
311
312 assert (rn->info);
313 oi = rn->info;
314
315 /* Call interface hook functions to clean up */
316 ospf_if_free (oi);
317
318 #ifdef HAVE_SNMP
319 ospf_snmp_if_update (c->ifp);
320 #endif /* HAVE_SNMP */
321
322 connected_free (c);
323
324 return 0;
325 }
326
327 void
328 ospf_zebra_add (struct prefix_ipv4 *p, struct ospf_route *or)
329 {
330 u_char message;
331 u_char distance;
332 u_char flags;
333 int psize;
334 struct stream *s;
335 struct ospf_path *path;
336 struct listnode *node;
337
338 if (zclient->redist[ZEBRA_ROUTE_OSPF])
339 {
340 message = 0;
341 flags = 0;
342
343 /* OSPF pass nexthop and metric */
344 SET_FLAG (message, ZAPI_MESSAGE_NEXTHOP);
345 SET_FLAG (message, ZAPI_MESSAGE_METRIC);
346
347 /* Distance value. */
348 distance = ospf_distance_apply (p, or);
349 if (distance)
350 SET_FLAG (message, ZAPI_MESSAGE_DISTANCE);
351
352 /* Make packet. */
353 s = zclient->obuf;
354 stream_reset (s);
355
356 /* Put command, type, flags, message. */
357 zclient_create_header (s, ZEBRA_IPV4_ROUTE_ADD);
358 stream_putc (s, ZEBRA_ROUTE_OSPF);
359 stream_putc (s, flags);
360 stream_putc (s, message);
361
362 /* Put prefix information. */
363 psize = PSIZE (p->prefixlen);
364 stream_putc (s, p->prefixlen);
365 stream_write (s, (u_char *) & p->prefix, psize);
366
367 /* Nexthop count. */
368 stream_putc (s, or->paths->count);
369
370 /* Nexthop, ifindex, distance and metric information. */
371 for (ALL_LIST_ELEMENTS_RO (or->paths, node, path))
372 {
373 if (path->nexthop.s_addr != INADDR_ANY)
374 {
375 stream_putc (s, ZEBRA_NEXTHOP_IPV4);
376 stream_put_in_addr (s, &path->nexthop);
377 }
378 else
379 {
380 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
381 if (path->ifindex)
382 stream_putl (s, path->ifindex);
383 else
384 stream_putl (s, 0);
385 }
386
387 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
388 {
389 char buf[2][INET_ADDRSTRLEN];
390 zlog_debug("Zebra: Route add %s/%d nexthop %s",
391 inet_ntop(AF_INET, &p->prefix,
392 buf[0], sizeof(buf[0])),
393 p->prefixlen,
394 inet_ntop(AF_INET, &path->nexthop,
395 buf[1], sizeof(buf[1])));
396 }
397 }
398
399 if (CHECK_FLAG (message, ZAPI_MESSAGE_DISTANCE))
400 stream_putc (s, distance);
401 if (CHECK_FLAG (message, ZAPI_MESSAGE_METRIC))
402 {
403 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL)
404 stream_putl (s, or->cost + or->u.ext.type2_cost);
405 else if (or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
406 stream_putl (s, or->u.ext.type2_cost);
407 else
408 stream_putl (s, or->cost);
409 }
410
411 stream_putw_at (s, 0, stream_get_endp (s));
412
413 zclient_send_message(zclient);
414 }
415 }
416
417 void
418 ospf_zebra_delete (struct prefix_ipv4 *p, struct ospf_route *or)
419 {
420 struct zapi_ipv4 api;
421 struct ospf_path *path;
422 struct in_addr *nexthop;
423 struct listnode *node, *nnode;
424
425 if (zclient->redist[ZEBRA_ROUTE_OSPF])
426 {
427 api.type = ZEBRA_ROUTE_OSPF;
428 api.flags = 0;
429 api.message = 0;
430 api.ifindex_num = 0;
431 api.nexthop_num = 0;
432
433 for (ALL_LIST_ELEMENTS (or->paths, node, nnode, path))
434 {
435 if (path->nexthop.s_addr != INADDR_ANY)
436 {
437 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
438 api.nexthop_num = 1;
439 nexthop = &path->nexthop;
440 api.nexthop = &nexthop;
441 }
442 else if (if_lookup_by_index(path->ifindex))
443 {
444 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
445 api.ifindex_num = 1;
446 api.ifindex = &path->ifindex;
447 }
448 else if ( IS_DEBUG_OSPF(zebra,ZEBRA_REDISTRIBUTE) )
449 {
450 zlog_debug("Zebra: no ifp %s %d",
451 inet_ntoa(p->prefix),
452 p->prefixlen);
453 }
454
455 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
456
457 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.nexthop_num)
458 {
459 char buf[2][INET_ADDRSTRLEN];
460 zlog_debug("Zebra: Route delete %s/%d nexthop %s",
461 inet_ntop(AF_INET, &p->prefix, buf[0], sizeof(buf[0])),
462 p->prefixlen,
463 inet_ntop(AF_INET, *api.nexthop,
464 buf[1], sizeof(buf[1])));
465 }
466 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.ifindex_num)
467 {
468 zlog_debug ("Zebra: Route delete %s/%d ifindex %d",
469 inet_ntoa (p->prefix),
470 p->prefixlen, *api.ifindex);
471 }
472 }
473 }
474 }
475
476 void
477 ospf_zebra_add_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
490 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_ADD, zclient, p, &api);
491
492 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
493 zlog_debug ("Zebra: Route add discard %s/%d",
494 inet_ntoa (p->prefix), p->prefixlen);
495 }
496 }
497
498 void
499 ospf_zebra_delete_discard (struct prefix_ipv4 *p)
500 {
501 struct zapi_ipv4 api;
502
503 if (zclient->redist[ZEBRA_ROUTE_OSPF])
504 {
505 api.type = ZEBRA_ROUTE_OSPF;
506 api.flags = ZEBRA_FLAG_BLACKHOLE;
507 api.message = 0;
508 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
509 api.nexthop_num = 0;
510 api.ifindex_num = 0;
511
512 zapi_ipv4_route (ZEBRA_IPV4_ROUTE_DELETE, zclient, p, &api);
513
514 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
515 zlog_debug ("Zebra: Route delete discard %s/%d",
516 inet_ntoa (p->prefix), p->prefixlen);
517
518 }
519 }
520
521 int
522 ospf_is_type_redistributed (int type)
523 {
524 return (DEFAULT_ROUTE_TYPE (type)) ?
525 zclient->default_information : zclient->redist[type];
526 }
527
528 int
529 ospf_redistribute_set (struct ospf *ospf, int type, int mtype, int mvalue)
530 {
531 int force = 0;
532
533 if (ospf_is_type_redistributed (type))
534 {
535 if (mtype != ospf->dmetric[type].type)
536 {
537 ospf->dmetric[type].type = mtype;
538 force = LSA_REFRESH_FORCE;
539 }
540 if (mvalue != ospf->dmetric[type].value)
541 {
542 ospf->dmetric[type].value = mvalue;
543 force = LSA_REFRESH_FORCE;
544 }
545
546 ospf_external_lsa_refresh_type (ospf, type, force);
547
548 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
549 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
550 ospf_redist_string(type),
551 metric_type (ospf, type), metric_value (ospf, type));
552
553 return CMD_SUCCESS;
554 }
555
556 ospf->dmetric[type].type = mtype;
557 ospf->dmetric[type].value = mvalue;
558
559 zclient_redistribute (ZEBRA_REDISTRIBUTE_ADD, zclient, type);
560
561 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
562 zlog_debug ("Redistribute[%s]: Start Type[%d], Metric[%d]",
563 ospf_redist_string(type),
564 metric_type (ospf, type), metric_value (ospf, type));
565
566 ospf_asbr_status_update (ospf, ++ospf->redistribute);
567
568 return CMD_SUCCESS;
569 }
570
571 int
572 ospf_redistribute_unset (struct ospf *ospf, int type)
573 {
574 if (type == zclient->redist_default)
575 return CMD_SUCCESS;
576
577 if (!ospf_is_type_redistributed (type))
578 return CMD_SUCCESS;
579
580 zclient_redistribute (ZEBRA_REDISTRIBUTE_DELETE, zclient, type);
581
582 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
583 zlog_debug ("Redistribute[%s]: Stop",
584 ospf_redist_string(type));
585
586 ospf->dmetric[type].type = -1;
587 ospf->dmetric[type].value = -1;
588
589 /* Remove the routes from OSPF table. */
590 ospf_redistribute_withdraw (ospf, type);
591
592 ospf_asbr_status_update (ospf, --ospf->redistribute);
593
594 return CMD_SUCCESS;
595 }
596
597 int
598 ospf_redistribute_default_set (struct ospf *ospf, int originate,
599 int mtype, int mvalue)
600 {
601 ospf->default_originate = originate;
602 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
603 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
604
605 if (ospf_is_type_redistributed (DEFAULT_ROUTE))
606 {
607 /* if ospf->default_originate changes value, is calling
608 ospf_external_lsa_refresh_default sufficient to implement
609 the change? */
610 ospf_external_lsa_refresh_default (ospf);
611
612 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
613 zlog_debug ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
614 ospf_redist_string(DEFAULT_ROUTE),
615 metric_type (ospf, DEFAULT_ROUTE),
616 metric_value (ospf, DEFAULT_ROUTE));
617 return CMD_SUCCESS;
618 }
619
620 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_ADD, zclient);
621
622 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
623 zlog_debug ("Redistribute[DEFAULT]: Start Type[%d], Metric[%d]",
624 metric_type (ospf, DEFAULT_ROUTE),
625 metric_value (ospf, DEFAULT_ROUTE));
626
627 if (ospf->router_id.s_addr == 0)
628 ospf->external_origin |= (1 << DEFAULT_ROUTE);
629 else
630 thread_add_timer (master, ospf_default_originate_timer, ospf, 1);
631
632 ospf_asbr_status_update (ospf, ++ospf->redistribute);
633
634 return CMD_SUCCESS;
635 }
636
637 int
638 ospf_redistribute_default_unset (struct ospf *ospf)
639 {
640 if (!ospf_is_type_redistributed (DEFAULT_ROUTE))
641 return CMD_SUCCESS;
642
643 ospf->default_originate = DEFAULT_ORIGINATE_NONE;
644 ospf->dmetric[DEFAULT_ROUTE].type = -1;
645 ospf->dmetric[DEFAULT_ROUTE].value = -1;
646
647 zclient_redistribute_default (ZEBRA_REDISTRIBUTE_DEFAULT_DELETE, zclient);
648
649 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
650 zlog_debug ("Redistribute[DEFAULT]: Stop");
651
652 ospf_asbr_status_update (ospf, --ospf->redistribute);
653
654 return CMD_SUCCESS;
655 }
656
657 static int
658 ospf_external_lsa_originate_check (struct ospf *ospf,
659 struct external_info *ei)
660 {
661 /* If prefix is multicast, then do not originate LSA. */
662 if (IN_MULTICAST (htonl (ei->p.prefix.s_addr)))
663 {
664 zlog_info ("LSA[Type5:%s]: Not originate AS-external-LSA, "
665 "Prefix belongs multicast", inet_ntoa (ei->p.prefix));
666 return 0;
667 }
668
669 /* Take care of default-originate. */
670 if (is_prefix_default (&ei->p))
671 if (ospf->default_originate == DEFAULT_ORIGINATE_NONE)
672 {
673 zlog_info ("LSA[Type5:0.0.0.0]: Not originate AS-exntenal-LSA "
674 "for default");
675 return 0;
676 }
677
678 return 1;
679 }
680
681 /* If connected prefix is OSPF enable interface, then do not announce. */
682 int
683 ospf_distribute_check_connected (struct ospf *ospf, struct external_info *ei)
684 {
685 struct listnode *node;
686 struct ospf_interface *oi;
687
688
689 for (ALL_LIST_ELEMENTS_RO (ospf->oiflist, node, oi))
690 if (prefix_match (oi->address, (struct prefix *) &ei->p))
691 return 0;
692 return 1;
693 }
694
695 /* return 1 if external LSA must be originated, 0 otherwise */
696 int
697 ospf_redistribute_check (struct ospf *ospf,
698 struct external_info *ei, int *changed)
699 {
700 struct route_map_set_values save_values;
701 struct prefix_ipv4 *p = &ei->p;
702 u_char type = is_prefix_default (&ei->p) ? DEFAULT_ROUTE : ei->type;
703
704 if (changed)
705 *changed = 0;
706
707 if (!ospf_external_lsa_originate_check (ospf, ei))
708 return 0;
709
710 /* Take care connected route. */
711 if (type == ZEBRA_ROUTE_CONNECT &&
712 !ospf_distribute_check_connected (ospf, ei))
713 return 0;
714
715 if (!DEFAULT_ROUTE_TYPE (type) && DISTRIBUTE_NAME (ospf, type))
716 /* distirbute-list exists, but access-list may not? */
717 if (DISTRIBUTE_LIST (ospf, type))
718 if (access_list_apply (DISTRIBUTE_LIST (ospf, type), p) == FILTER_DENY)
719 {
720 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
721 zlog_debug ("Redistribute[%s]: %s/%d filtered by ditribute-list.",
722 ospf_redist_string(type),
723 inet_ntoa (p->prefix), p->prefixlen);
724 return 0;
725 }
726
727 save_values = ei->route_map_set;
728 ospf_reset_route_map_set_values (&ei->route_map_set);
729
730 /* apply route-map if needed */
731 if (ROUTEMAP_NAME (ospf, type))
732 {
733 int ret;
734
735 ret = route_map_apply (ROUTEMAP (ospf, type), (struct prefix *) p,
736 RMAP_OSPF, ei);
737
738 if (ret == RMAP_DENYMATCH)
739 {
740 ei->route_map_set = save_values;
741 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
742 zlog_debug ("Redistribute[%s]: %s/%d filtered by route-map.",
743 ospf_redist_string(type),
744 inet_ntoa (p->prefix), p->prefixlen);
745 return 0;
746 }
747
748 /* check if 'route-map set' changed something */
749 if (changed)
750 *changed = !ospf_route_map_set_compare (&ei->route_map_set,
751 &save_values);
752 }
753
754 return 1;
755 }
756
757 /* OSPF route-map set for redistribution */
758 void
759 ospf_routemap_set (struct ospf *ospf, int type, const char *name)
760 {
761 if (ROUTEMAP_NAME (ospf, type))
762 free (ROUTEMAP_NAME (ospf, type));
763
764 ROUTEMAP_NAME (ospf, type) = strdup (name);
765 ROUTEMAP (ospf, type) = route_map_lookup_by_name (name);
766 }
767
768 void
769 ospf_routemap_unset (struct ospf *ospf, int type)
770 {
771 if (ROUTEMAP_NAME (ospf, type))
772 free (ROUTEMAP_NAME (ospf, type));
773
774 ROUTEMAP_NAME (ospf, type) = NULL;
775 ROUTEMAP (ospf, type) = NULL;
776 }
777
778 /* Zebra route add and delete treatment. */
779 static int
780 ospf_zebra_read_ipv4 (int command, struct zclient *zclient,
781 zebra_size_t length)
782 {
783 struct stream *s;
784 struct zapi_ipv4 api;
785 unsigned long ifindex;
786 struct in_addr nexthop;
787 struct prefix_ipv4 p;
788 struct external_info *ei;
789 struct ospf *ospf;
790
791 s = zclient->ibuf;
792 ifindex = 0;
793 nexthop.s_addr = 0;
794
795 /* Type, flags, message. */
796 api.type = stream_getc (s);
797 api.flags = stream_getc (s);
798 api.message = stream_getc (s);
799
800 /* IPv4 prefix. */
801 memset (&p, 0, sizeof (struct prefix_ipv4));
802 p.family = AF_INET;
803 p.prefixlen = stream_getc (s);
804 stream_get (&p.prefix, s, PSIZE (p.prefixlen));
805
806 if (IPV4_NET127(ntohl(p.prefix.s_addr)))
807 return 0;
808
809 /* Nexthop, ifindex, distance, metric. */
810 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP))
811 {
812 api.nexthop_num = stream_getc (s);
813 nexthop.s_addr = stream_get_ipv4 (s);
814 }
815 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_IFINDEX))
816 {
817 api.ifindex_num = stream_getc (s);
818 /* XXX assert(api.ifindex_num == 1); */
819 ifindex = stream_getl (s);
820 }
821 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_DISTANCE))
822 api.distance = stream_getc (s);
823 if (CHECK_FLAG (api.message, ZAPI_MESSAGE_METRIC))
824 api.metric = stream_getl (s);
825
826 ospf = ospf_lookup ();
827 if (ospf == NULL)
828 return 0;
829
830 if (command == ZEBRA_IPV4_ROUTE_ADD)
831 {
832 /* XXX|HACK|TODO|FIXME:
833 * Maybe we should ignore reject/blackhole routes? Testing shows that
834 * there is no problems though and this is only way to "summarize"
835 * routes in ASBR at the moment. Maybe we need just a better generalised
836 * solution for these types?
837 *
838 * if ( CHECK_FLAG (api.flags, ZEBRA_FLAG_BLACKHOLE)
839 * || CHECK_FLAG (api.flags, ZEBRA_FLAG_REJECT))
840 * return 0;
841 */
842
843 ei = ospf_external_info_add (api.type, p, ifindex, nexthop);
844
845 if (ospf->router_id.s_addr == 0)
846 /* Set flags to generate AS-external-LSA originate event
847 for each redistributed protocols later. */
848 ospf->external_origin |= (1 << api.type);
849 else
850 {
851 if (ei)
852 {
853 if (is_prefix_default (&p))
854 ospf_external_lsa_refresh_default (ospf);
855 else
856 {
857 struct ospf_lsa *current;
858
859 current = ospf_external_info_find_lsa (ospf, &ei->p);
860 if (!current)
861 ospf_external_lsa_originate (ospf, ei);
862 else if (IS_LSA_MAXAGE (current))
863 ospf_external_lsa_refresh (ospf, current,
864 ei, LSA_REFRESH_FORCE);
865 else
866 zlog_warn ("ospf_zebra_read_ipv4() : %s already exists",
867 inet_ntoa (p.prefix));
868 }
869 }
870 }
871 }
872 else /* if (command == ZEBRA_IPV4_ROUTE_DELETE) */
873 {
874 ospf_external_info_delete (api.type, p);
875 if (is_prefix_default (&p))
876 ospf_external_lsa_refresh_default (ospf);
877 else
878 ospf_external_lsa_flush (ospf, api.type, &p, ifindex /*, nexthop */);
879 }
880
881 return 0;
882 }
883 \f
884
885 int
886 ospf_distribute_list_out_set (struct ospf *ospf, int type, const char *name)
887 {
888 /* Lookup access-list for distribute-list. */
889 DISTRIBUTE_LIST (ospf, type) = access_list_lookup (AFI_IP, name);
890
891 /* Clear previous distribute-name. */
892 if (DISTRIBUTE_NAME (ospf, type))
893 free (DISTRIBUTE_NAME (ospf, type));
894
895 /* Set distribute-name. */
896 DISTRIBUTE_NAME (ospf, type) = strdup (name);
897
898 /* If access-list have been set, schedule update timer. */
899 if (DISTRIBUTE_LIST (ospf, type))
900 ospf_distribute_list_update (ospf, type);
901
902 return CMD_SUCCESS;
903 }
904
905 int
906 ospf_distribute_list_out_unset (struct ospf *ospf, int type, const char *name)
907 {
908 /* Schedule update timer. */
909 if (DISTRIBUTE_LIST (ospf, type))
910 ospf_distribute_list_update (ospf, type);
911
912 /* Unset distribute-list. */
913 DISTRIBUTE_LIST (ospf, type) = NULL;
914
915 /* Clear distribute-name. */
916 if (DISTRIBUTE_NAME (ospf, type))
917 free (DISTRIBUTE_NAME (ospf, type));
918
919 DISTRIBUTE_NAME (ospf, type) = NULL;
920
921 return CMD_SUCCESS;
922 }
923
924 /* distribute-list update timer. */
925 static int
926 ospf_distribute_list_update_timer (struct thread *thread)
927 {
928 struct route_node *rn;
929 struct external_info *ei;
930 struct route_table *rt;
931 struct ospf_lsa *lsa;
932 int type;
933 struct ospf *ospf;
934
935 ospf = ospf_lookup ();
936 if (ospf == NULL)
937 return 0;
938
939 ospf->t_distribute_update = NULL;
940
941 zlog_info ("Zebra[Redistribute]: distribute-list update timer fired!");
942
943 /* foreach all external info. */
944 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
945 {
946 rt = EXTERNAL_INFO (type);
947 if (!rt)
948 continue;
949 for (rn = route_top (rt); rn; rn = route_next (rn))
950 if ((ei = rn->info) != NULL)
951 {
952 if (is_prefix_default (&ei->p))
953 ospf_external_lsa_refresh_default (ospf);
954 else if ((lsa = ospf_external_info_find_lsa (ospf, &ei->p)))
955 ospf_external_lsa_refresh (ospf, lsa, ei, LSA_REFRESH_IF_CHANGED);
956 else
957 ospf_external_lsa_originate (ospf, ei);
958 }
959 }
960 return 0;
961 }
962
963 #define OSPF_DISTRIBUTE_UPDATE_DELAY 5
964
965 /* Update distribute-list and set timer to apply access-list. */
966 void
967 ospf_distribute_list_update (struct ospf *ospf, int type)
968 {
969 struct route_table *rt;
970
971 /* External info does not exist. */
972 if (!(rt = EXTERNAL_INFO (type)))
973 return;
974
975 /* If exists previously invoked thread, then let it continue. */
976 if (ospf->t_distribute_update)
977 return;
978
979 /* Set timer. */
980 ospf->t_distribute_update =
981 thread_add_timer (master, ospf_distribute_list_update_timer,
982 (void *) type, OSPF_DISTRIBUTE_UPDATE_DELAY);
983 }
984
985 /* If access-list is updated, apply some check. */
986 static void
987 ospf_filter_update (struct access_list *access)
988 {
989 struct ospf *ospf;
990 int type;
991 int abr_inv = 0;
992 struct ospf_area *area;
993 struct listnode *node;
994
995 /* If OSPF instatnce does not exist, return right now. */
996 ospf = ospf_lookup ();
997 if (ospf == NULL)
998 return;
999
1000 /* Update distribute-list, and apply filter. */
1001 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
1002 {
1003 if (ROUTEMAP (ospf, type) != NULL)
1004 {
1005 /* if route-map is not NULL it may be using this access list */
1006 ospf_distribute_list_update (ospf, type);
1007 continue;
1008 }
1009
1010 /* There is place for route-map for default-information (ZEBRA_ROUTE_MAX),
1011 * but no distribute list. */
1012 if (type == ZEBRA_ROUTE_MAX)
1013 break;
1014
1015 if (DISTRIBUTE_NAME (ospf, type))
1016 {
1017 /* Keep old access-list for distribute-list. */
1018 struct access_list *old = DISTRIBUTE_LIST (ospf, type);
1019
1020 /* Update access-list for distribute-list. */
1021 DISTRIBUTE_LIST (ospf, type) =
1022 access_list_lookup (AFI_IP, DISTRIBUTE_NAME (ospf, type));
1023
1024 /* No update for this distribute type. */
1025 if (old == NULL && DISTRIBUTE_LIST (ospf, type) == NULL)
1026 continue;
1027
1028 /* Schedule distribute-list update timer. */
1029 if (DISTRIBUTE_LIST (ospf, type) == NULL ||
1030 strcmp (DISTRIBUTE_NAME (ospf, type), access->name) == 0)
1031 ospf_distribute_list_update (ospf, type);
1032 }
1033 }
1034
1035 /* Update Area access-list. */
1036 for (ALL_LIST_ELEMENTS_RO (ospf->areas, node, area))
1037 {
1038 if (EXPORT_NAME (area))
1039 {
1040 EXPORT_LIST (area) = NULL;
1041 abr_inv++;
1042 }
1043
1044 if (IMPORT_NAME (area))
1045 {
1046 IMPORT_LIST (area) = NULL;
1047 abr_inv++;
1048 }
1049 }
1050
1051 /* Schedule ABR tasks -- this will be changed -- takada. */
1052 if (IS_OSPF_ABR (ospf) && abr_inv)
1053 ospf_schedule_abr_task (ospf);
1054 }
1055
1056 /* If prefix-list is updated, do some updates. */
1057 void
1058 ospf_prefix_list_update (struct prefix_list *plist)
1059 {
1060 struct ospf *ospf;
1061 int type;
1062 int abr_inv = 0;
1063 struct ospf_area *area;
1064 struct listnode *node;
1065
1066 /* If OSPF instatnce does not exist, return right now. */
1067 ospf = ospf_lookup ();
1068 if (ospf == NULL)
1069 return;
1070
1071 /* Update all route-maps which are used as redistribution filters.
1072 * They might use prefix-list.
1073 */
1074 for (type = 0; type <= ZEBRA_ROUTE_MAX; type++)
1075 {
1076 if (ROUTEMAP (ospf, type) != NULL)
1077 {
1078 /* If route-map is not NULL it may be using this prefix list */
1079 ospf_distribute_list_update (ospf, type);
1080 continue;
1081 }
1082 }
1083
1084 /* Update area filter-lists. */
1085 for (ALL_LIST_ELEMENTS_RO (ospf->areas, node, area))
1086 {
1087 /* Update filter-list in. */
1088 if (PREFIX_NAME_IN (area))
1089 if (strcmp (PREFIX_NAME_IN (area), plist->name) == 0)
1090 {
1091 PREFIX_LIST_IN (area) =
1092 prefix_list_lookup (AFI_IP, PREFIX_NAME_IN (area));
1093 abr_inv++;
1094 }
1095
1096 /* Update filter-list out. */
1097 if (PREFIX_NAME_OUT (area))
1098 if (strcmp (PREFIX_NAME_OUT (area), plist->name) == 0)
1099 {
1100 PREFIX_LIST_IN (area) =
1101 prefix_list_lookup (AFI_IP, PREFIX_NAME_OUT (area));
1102 abr_inv++;
1103 }
1104 }
1105
1106 /* Schedule ABR task. */
1107 if (IS_OSPF_ABR (ospf) && abr_inv)
1108 ospf_schedule_abr_task (ospf);
1109 }
1110
1111 static struct ospf_distance *
1112 ospf_distance_new (void)
1113 {
1114 return XCALLOC (MTYPE_OSPF_DISTANCE, sizeof (struct ospf_distance));
1115 }
1116
1117 static void
1118 ospf_distance_free (struct ospf_distance *odistance)
1119 {
1120 XFREE (MTYPE_OSPF_DISTANCE, odistance);
1121 }
1122
1123 int
1124 ospf_distance_set (struct vty *vty, struct ospf *ospf,
1125 const char *distance_str,
1126 const char *ip_str,
1127 const char *access_list_str)
1128 {
1129 int ret;
1130 struct prefix_ipv4 p;
1131 u_char distance;
1132 struct route_node *rn;
1133 struct ospf_distance *odistance;
1134
1135 ret = str2prefix_ipv4 (ip_str, &p);
1136 if (ret == 0)
1137 {
1138 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1139 return CMD_WARNING;
1140 }
1141
1142 distance = atoi (distance_str);
1143
1144 /* Get OSPF distance node. */
1145 rn = route_node_get (ospf->distance_table, (struct prefix *) &p);
1146 if (rn->info)
1147 {
1148 odistance = rn->info;
1149 route_unlock_node (rn);
1150 }
1151 else
1152 {
1153 odistance = ospf_distance_new ();
1154 rn->info = odistance;
1155 }
1156
1157 /* Set distance value. */
1158 odistance->distance = distance;
1159
1160 /* Reset access-list configuration. */
1161 if (odistance->access_list)
1162 {
1163 free (odistance->access_list);
1164 odistance->access_list = NULL;
1165 }
1166 if (access_list_str)
1167 odistance->access_list = strdup (access_list_str);
1168
1169 return CMD_SUCCESS;
1170 }
1171
1172 int
1173 ospf_distance_unset (struct vty *vty, struct ospf *ospf,
1174 const char *distance_str,
1175 const char *ip_str, char
1176 const *access_list_str)
1177 {
1178 int ret;
1179 struct prefix_ipv4 p;
1180 u_char distance;
1181 struct route_node *rn;
1182 struct ospf_distance *odistance;
1183
1184 ret = str2prefix_ipv4 (ip_str, &p);
1185 if (ret == 0)
1186 {
1187 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1188 return CMD_WARNING;
1189 }
1190
1191 distance = atoi (distance_str);
1192
1193 rn = route_node_lookup (ospf->distance_table, (struct prefix *) &p);
1194 if (!rn)
1195 {
1196 vty_out (vty, "Can't find specified prefix%s", VTY_NEWLINE);
1197 return CMD_WARNING;
1198 }
1199
1200 odistance = rn->info;
1201
1202 if (odistance->access_list)
1203 free (odistance->access_list);
1204 ospf_distance_free (odistance);
1205
1206 rn->info = NULL;
1207 route_unlock_node (rn);
1208 route_unlock_node (rn);
1209
1210 return CMD_SUCCESS;
1211 }
1212
1213 void
1214 ospf_distance_reset (struct ospf *ospf)
1215 {
1216 struct route_node *rn;
1217 struct ospf_distance *odistance;
1218
1219 for (rn = route_top (ospf->distance_table); rn; rn = route_next (rn))
1220 if ((odistance = rn->info) != NULL)
1221 {
1222 if (odistance->access_list)
1223 free (odistance->access_list);
1224 ospf_distance_free (odistance);
1225 rn->info = NULL;
1226 route_unlock_node (rn);
1227 }
1228 }
1229
1230 u_char
1231 ospf_distance_apply (struct prefix_ipv4 *p, struct ospf_route *or)
1232 {
1233 struct ospf *ospf;
1234
1235 ospf = ospf_lookup ();
1236 if (ospf == NULL)
1237 return 0;
1238
1239 if (ospf->distance_intra)
1240 if (or->path_type == OSPF_PATH_INTRA_AREA)
1241 return ospf->distance_intra;
1242
1243 if (ospf->distance_inter)
1244 if (or->path_type == OSPF_PATH_INTER_AREA)
1245 return ospf->distance_inter;
1246
1247 if (ospf->distance_external)
1248 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL
1249 || or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
1250 return ospf->distance_external;
1251
1252 if (ospf->distance_all)
1253 return ospf->distance_all;
1254
1255 return 0;
1256 }
1257
1258 void
1259 ospf_zebra_init ()
1260 {
1261 /* Allocate zebra structure. */
1262 zclient = zclient_new ();
1263 zclient_init (zclient, ZEBRA_ROUTE_OSPF);
1264 zclient->router_id_update = ospf_router_id_update_zebra;
1265 zclient->interface_add = ospf_interface_add;
1266 zclient->interface_delete = ospf_interface_delete;
1267 zclient->interface_up = ospf_interface_state_up;
1268 zclient->interface_down = ospf_interface_state_down;
1269 zclient->interface_address_add = ospf_interface_address_add;
1270 zclient->interface_address_delete = ospf_interface_address_delete;
1271 zclient->ipv4_route_add = ospf_zebra_read_ipv4;
1272 zclient->ipv4_route_delete = ospf_zebra_read_ipv4;
1273
1274 access_list_add_hook (ospf_filter_update);
1275 access_list_delete_hook (ospf_filter_update);
1276 prefix_list_add_hook (ospf_prefix_list_update);
1277 prefix_list_delete_hook (ospf_prefix_list_update);
1278 }