blob: 300105f492b1997be2ef5b9f515b84b48b6ac8aa [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"
35#include "log.h"
36
37#include "ospfd/ospfd.h"
38#include "ospfd/ospf_interface.h"
39#include "ospfd/ospf_ism.h"
40#include "ospfd/ospf_asbr.h"
41#include "ospfd/ospf_asbr.h"
42#include "ospfd/ospf_abr.h"
43#include "ospfd/ospf_lsa.h"
44#include "ospfd/ospf_dump.h"
45#include "ospfd/ospf_route.h"
46#include "ospfd/ospf_zebra.h"
47#ifdef HAVE_SNMP
48#include "ospfd/ospf_snmp.h"
49#endif /* HAVE_SNMP */
50
51/* Zebra structure to hold current status. */
52struct zclient *zclient = NULL;
53
54/* For registering threads. */
55extern struct thread_master *master;
56
57/* Inteface addition message from zebra. */
58int
59ospf_interface_add (int command, struct zclient *zclient, zebra_size_t length)
60{
61 struct interface *ifp;
paul020709f2003-04-04 02:44:16 +000062 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +000063
64 ifp = zebra_interface_add_read (zclient->ibuf);
65
66 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
67 zlog_info ("Zebra: interface add %s index %d flags %ld metric %d mtu %d",
paulcf795c52003-06-19 02:13:25 +000068 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +000069
paulcf795c52003-06-19 02:13:25 +000070 assert (ifp->info);
paulf2c80652002-12-13 21:44:27 +000071
paul718e3742002-12-13 20:15:29 +000072 if (!OSPF_IF_PARAM_CONFIGURED (IF_DEF_PARAMS (ifp), type))
73 {
74 SET_IF_PARAM (IF_DEF_PARAMS (ifp), type);
75 IF_DEF_PARAMS (ifp)->type = OSPF_IFTYPE_BROADCAST;
paulcf795c52003-06-19 02:13:25 +000076
paul718e3742002-12-13 20:15:29 +000077 if (if_is_broadcast (ifp))
paulcf795c52003-06-19 02:13:25 +000078 IF_DEF_PARAMS (ifp)->type = OSPF_IFTYPE_BROADCAST;
paul718e3742002-12-13 20:15:29 +000079 else if (if_is_pointopoint (ifp))
paulcf795c52003-06-19 02:13:25 +000080 IF_DEF_PARAMS (ifp)->type = OSPF_IFTYPE_POINTOPOINT;
paul718e3742002-12-13 20:15:29 +000081 else if (if_is_loopback (ifp))
paulcf795c52003-06-19 02:13:25 +000082 IF_DEF_PARAMS (ifp)->type = OSPF_IFTYPE_LOOPBACK;
paul718e3742002-12-13 20:15:29 +000083 }
84
paul020709f2003-04-04 02:44:16 +000085 ospf = ospf_lookup ();
86 if (ospf != NULL)
87 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +000088
89#ifdef HAVE_SNMP
90 ospf_snmp_if_update (ifp);
91#endif /* HAVE_SNMP */
92
93 return 0;
94}
95
96int
97ospf_interface_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +000098 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +000099{
100 struct interface *ifp;
101 struct stream *s;
102 struct route_node *rn;
103
paulcf795c52003-06-19 02:13:25 +0000104 s = zclient->ibuf;
paul718e3742002-12-13 20:15:29 +0000105 /* zebra_interface_state_read() updates interface structure in iflist */
106 ifp = zebra_interface_state_read (s);
107
108 if (ifp == NULL)
109 return 0;
110
111 if (if_is_up (ifp))
112 zlog_warn ("Zebra: got delete of %s, but interface is still up",
paulcf795c52003-06-19 02:13:25 +0000113 ifp->name);
114
paul718e3742002-12-13 20:15:29 +0000115 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
paulcf795c52003-06-19 02:13:25 +0000116 zlog_info
117 ("Zebra: interface delete %s index %d flags %ld metric %d mtu %d",
118 ifp->name, ifp->ifindex, ifp->flags, ifp->metric, ifp->mtu);
paul718e3742002-12-13 20:15:29 +0000119
120#ifdef HAVE_SNMP
121 ospf_snmp_if_delete (ifp);
122#endif /* HAVE_SNMP */
123
124 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
125 if (rn->info)
126 ospf_if_free ((struct ospf_interface *) rn->info);
127
paul718e3742002-12-13 20:15:29 +0000128 return 0;
129}
130
131struct interface *
132zebra_interface_if_lookup (struct stream *s)
133{
134 struct interface *ifp;
135 u_char ifname_tmp[INTERFACE_NAMSIZ];
136
137 /* Read interface name. */
138 stream_get (ifname_tmp, s, INTERFACE_NAMSIZ);
139
140 /* Lookup this by interface index. */
141 ifp = if_lookup_by_name (ifname_tmp);
142
143 /* If such interface does not exist, indicate an error */
144 if (!ifp)
145 return NULL;
146
147 return ifp;
148}
149
150void
151zebra_interface_if_set_value (struct stream *s, struct interface *ifp)
152{
153 /* Read interface's index. */
154 ifp->ifindex = stream_getl (s);
155
156 /* Read interface's value. */
paul2e3b2e42002-12-13 21:03:13 +0000157 ifp->status = stream_getc (s);
paul718e3742002-12-13 20:15:29 +0000158 ifp->flags = stream_getl (s);
159 ifp->metric = stream_getl (s);
160 ifp->mtu = stream_getl (s);
161 ifp->bandwidth = stream_getl (s);
162}
163
164int
165ospf_interface_state_up (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000166 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000167{
168 struct interface *ifp;
169 struct interface if_tmp;
170 struct ospf_interface *oi;
171 struct route_node *rn;
paulcf795c52003-06-19 02:13:25 +0000172
paul718e3742002-12-13 20:15:29 +0000173 ifp = zebra_interface_if_lookup (zclient->ibuf);
174
175 if (ifp == NULL)
176 return 0;
177
178 /* Interface is already up. */
paul2e3b2e42002-12-13 21:03:13 +0000179 if (if_is_operative (ifp))
paul718e3742002-12-13 20:15:29 +0000180 {
181 /* Temporarily keep ifp values. */
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))
paulcf795c52003-06-19 02:13:25 +0000187 zlog_info ("Zebra: Interface[%s] state update.", ifp->name);
paul718e3742002-12-13 20:15:29 +0000188
189 if (if_tmp.bandwidth != ifp->bandwidth)
paulcf795c52003-06-19 02:13:25 +0000190 {
191 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
192 zlog_info ("Zebra: Interface[%s] bandwidth change %d -> %d.",
193 ifp->name, if_tmp.bandwidth, ifp->bandwidth);
paul718e3742002-12-13 20:15:29 +0000194
paulcf795c52003-06-19 02:13:25 +0000195 ospf_if_recalculate_output_cost (ifp);
196 }
paul718e3742002-12-13 20:15:29 +0000197 return 0;
198 }
paulcf795c52003-06-19 02:13:25 +0000199
paul718e3742002-12-13 20:15:29 +0000200 zebra_interface_if_set_value (zclient->ibuf, ifp);
paulcf795c52003-06-19 02:13:25 +0000201
paul718e3742002-12-13 20:15:29 +0000202 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
203 zlog_info ("Zebra: Interface[%s] state change to up.", ifp->name);
paulcf795c52003-06-19 02:13:25 +0000204
205 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +0000206 {
paulcf795c52003-06-19 02:13:25 +0000207 if ((oi = rn->info) == NULL)
208 continue;
209
paul718e3742002-12-13 20:15:29 +0000210 ospf_if_up (oi);
211 }
paulcf795c52003-06-19 02:13:25 +0000212
paul718e3742002-12-13 20:15:29 +0000213 return 0;
214}
215
216int
217ospf_interface_state_down (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000218 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000219{
220 struct interface *ifp;
221 struct ospf_interface *oi;
222 struct route_node *node;
223
224 ifp = zebra_interface_state_read (zclient->ibuf);
225
226 if (ifp == NULL)
227 return 0;
228
229 if (IS_DEBUG_OSPF (zebra, ZEBRA_INTERFACE))
230 zlog_info ("Zebra: Interface[%s] state change to down.", ifp->name);
231
paulcf795c52003-06-19 02:13:25 +0000232 for (node = route_top (IF_OIFS (ifp)); node; node = route_next (node))
paul718e3742002-12-13 20:15:29 +0000233 {
paulcf795c52003-06-19 02:13:25 +0000234 if ((oi = node->info) == NULL)
235 continue;
paul718e3742002-12-13 20:15:29 +0000236 ospf_if_down (oi);
237 }
238
239 return 0;
240}
241
242int
243ospf_interface_address_add (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000244 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000245{
paul020709f2003-04-04 02:44:16 +0000246 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000247 struct connected *c;
248
249 c = zebra_interface_address_add_read (zclient->ibuf);
250
251 if (c == NULL)
252 return 0;
253
paul020709f2003-04-04 02:44:16 +0000254 ospf = ospf_lookup ();
255 if (ospf != NULL)
256 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000257
258#ifdef HAVE_SNMP
259 ospf_snmp_if_update (c->ifp);
260#endif /* HAVE_SNMP */
261
262 return 0;
263}
264
265int
266ospf_interface_address_delete (int command, struct zclient *zclient,
paulcf795c52003-06-19 02:13:25 +0000267 zebra_size_t length)
paul718e3742002-12-13 20:15:29 +0000268{
paul020709f2003-04-04 02:44:16 +0000269 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000270 struct connected *c;
271 struct interface *ifp;
272 struct ospf_interface *oi;
273 struct route_node *rn;
274 struct prefix p;
275
276 c = zebra_interface_address_delete_read (zclient->ibuf);
277
278 if (c == NULL)
279 return 0;
280
281 ifp = c->ifp;
282 p = *c->address;
283 p.prefixlen = IPV4_MAX_PREFIXLEN;
284
285 rn = route_node_lookup (IF_OIFS (ifp), &p);
paulcf795c52003-06-19 02:13:25 +0000286 if (!rn)
paul718e3742002-12-13 20:15:29 +0000287 return 0;
288
289 assert (rn->info);
290 oi = rn->info;
paulcf795c52003-06-19 02:13:25 +0000291
paul718e3742002-12-13 20:15:29 +0000292 /* Call interface hook functions to clean up */
293 ospf_if_free (oi);
paulcf795c52003-06-19 02:13:25 +0000294
paul718e3742002-12-13 20:15:29 +0000295#ifdef HAVE_SNMP
296 ospf_snmp_if_update (c->ifp);
297#endif /* HAVE_SNMP */
298
299 connected_free (c);
300
paul020709f2003-04-04 02:44:16 +0000301 ospf = ospf_lookup ();
302 if (ospf != NULL)
303 ospf_if_update (ospf);
paul718e3742002-12-13 20:15:29 +0000304
305 return 0;
306}
paul72357f22003-06-19 02:11:23 +0000307
paul718e3742002-12-13 20:15:29 +0000308void
309ospf_zebra_add (struct prefix_ipv4 *p, struct ospf_route *or)
310{
311 u_char message;
312 u_char distance;
313 u_char flags;
314 int psize;
315 struct stream *s;
316 struct ospf_path *path;
317 listnode node;
318
319 if (zclient->redist[ZEBRA_ROUTE_OSPF])
320 {
321 message = 0;
322 flags = 0;
323
324 /* OSPF pass nexthop and metric */
325 SET_FLAG (message, ZAPI_MESSAGE_NEXTHOP);
326 SET_FLAG (message, ZAPI_MESSAGE_METRIC);
327
328 /* Distance value. */
329 distance = ospf_distance_apply (p, or);
330 if (distance)
paul72357f22003-06-19 02:11:23 +0000331 SET_FLAG (message, ZAPI_MESSAGE_DISTANCE);
paul718e3742002-12-13 20:15:29 +0000332
333 /* Make packet. */
334 s = zclient->obuf;
335 stream_reset (s);
336
337 /* Length place holder. */
338 stream_putw (s, 0);
339
340 /* Put command, type, flags, message. */
341 stream_putc (s, ZEBRA_IPV4_ROUTE_ADD);
342 stream_putc (s, ZEBRA_ROUTE_OSPF);
343 stream_putc (s, flags);
344 stream_putc (s, message);
paulcf795c52003-06-19 02:13:25 +0000345
paul718e3742002-12-13 20:15:29 +0000346 /* Put prefix information. */
347 psize = PSIZE (p->prefixlen);
348 stream_putc (s, p->prefixlen);
paulcf795c52003-06-19 02:13:25 +0000349 stream_write (s, (u_char *) & p->prefix, psize);
paul718e3742002-12-13 20:15:29 +0000350
351 /* Nexthop count. */
paul96735ee2003-08-10 02:51:22 +0000352 stream_putc (s, or->paths->count);
paul718e3742002-12-13 20:15:29 +0000353
354 /* Nexthop, ifindex, distance and metric information. */
paul96735ee2003-08-10 02:51:22 +0000355 for (node = listhead (or->paths); node; nextnode (node))
paul72357f22003-06-19 02:11:23 +0000356 {
paulcf795c52003-06-19 02:13:25 +0000357 path = getdata (node);
paul718e3742002-12-13 20:15:29 +0000358
paulcf795c52003-06-19 02:13:25 +0000359 if (path->nexthop.s_addr != INADDR_ANY)
360 {
361 stream_putc (s, ZEBRA_NEXTHOP_IPV4);
362 stream_put_in_addr (s, &path->nexthop);
363 }
364 else
365 {
366 stream_putc (s, ZEBRA_NEXTHOP_IFINDEX);
367 if (path->oi)
368 stream_putl (s, path->oi->ifp->ifindex);
369 else
370 stream_putl (s, 0);
371 }
paul72357f22003-06-19 02:11:23 +0000372
373 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
374 {
375 zlog_info ("Zebra: Route add %s/%d nexthop %s",
paulcf795c52003-06-19 02:13:25 +0000376 inet_ntoa (p->prefix),
377 p->prefixlen, inet_ntoa (path->nexthop));
378 }
379 }
paul718e3742002-12-13 20:15:29 +0000380
381 if (CHECK_FLAG (message, ZAPI_MESSAGE_DISTANCE))
paulcf795c52003-06-19 02:13:25 +0000382 stream_putc (s, distance);
paul718e3742002-12-13 20:15:29 +0000383 if (CHECK_FLAG (message, ZAPI_MESSAGE_METRIC))
paulcf795c52003-06-19 02:13:25 +0000384 {
385 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL)
386 stream_putl (s, or->cost + or->u.ext.type2_cost);
387 else if (or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
388 stream_putl (s, or->u.ext.type2_cost);
389 else
390 stream_putl (s, or->cost);
391 }
paul718e3742002-12-13 20:15:29 +0000392
393 stream_putw_at (s, 0, stream_get_endp (s));
394
395 writen (zclient->sock, s->data, stream_get_endp (s));
paul718e3742002-12-13 20:15:29 +0000396 }
397}
398
399void
400ospf_zebra_delete (struct prefix_ipv4 *p, struct ospf_route *or)
401{
402 struct zapi_ipv4 api;
paul72357f22003-06-19 02:11:23 +0000403 struct ospf_path *path;
404 struct in_addr *nexthop;
405 listnode node;
paul718e3742002-12-13 20:15:29 +0000406
407 if (zclient->redist[ZEBRA_ROUTE_OSPF])
408 {
409 api.type = ZEBRA_ROUTE_OSPF;
410 api.flags = 0;
411 api.message = 0;
paul72357f22003-06-19 02:11:23 +0000412 api.ifindex_num = 0;
413 api.nexthop_num = 0;
paul718e3742002-12-13 20:15:29 +0000414
paul96735ee2003-08-10 02:51:22 +0000415 for (node = listhead (or->paths); node; nextnode (node))
paulcf795c52003-06-19 02:13:25 +0000416 {
417 path = getdata (node);
paul718e3742002-12-13 20:15:29 +0000418
paulcf795c52003-06-19 02:13:25 +0000419 if (path->nexthop.s_addr != INADDR_ANY)
420 {
pauld8e1d6b2003-08-10 04:04:41 +0000421 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
paulcf795c52003-06-19 02:13:25 +0000422 api.nexthop_num = 1;
423 nexthop = &path->nexthop;
424 api.nexthop = &nexthop;
425 }
hasso2db3d052004-02-11 21:52:13 +0000426 else if (ospf_if_exists(path->oi) && (path->oi->ifp))
paulbb8ff1e2003-08-12 06:00:30 +0000427 {
428 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
429 api.ifindex_num = 1;
430 api.ifindex = &path->oi->ifp->ifindex;
431 }
432 else if ( IS_DEBUG_OSPF(zebra,ZEBRA_REDISTRIBUTE) )
433 {
434 zlog_info("Zebra: no ifp %s %d",
435 inet_ntoa(p->prefix),
436 p->prefixlen);
437 }
paul72357f22003-06-19 02:11:23 +0000438
439 zapi_ipv4_delete (zclient, p, &api);
440
441 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.nexthop_num)
paulcf795c52003-06-19 02:13:25 +0000442 {
443 zlog_info ("Zebra: Route delete %s/%d nexthop %s",
444 inet_ntoa (p->prefix),
445 p->prefixlen, inet_ntoa (**api.nexthop));
446 }
paulbb8ff1e2003-08-12 06:00:30 +0000447 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE) && api.ifindex_num)
448 {
449 zlog_info ("Zebra: Route delete %s/%d ifindex %d",
450 inet_ntoa (p->prefix),
451 p->prefixlen, *api.ifindex);
452 }
paulcf795c52003-06-19 02:13:25 +0000453 }
paul718e3742002-12-13 20:15:29 +0000454 }
455}
456
457void
458ospf_zebra_add_discard (struct prefix_ipv4 *p)
459{
460 struct zapi_ipv4 api;
461
462 if (zclient->redist[ZEBRA_ROUTE_OSPF])
463 {
464 api.type = ZEBRA_ROUTE_OSPF;
465 api.flags = ZEBRA_FLAG_BLACKHOLE;
466 api.message = 0;
467 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
468 api.nexthop_num = 0;
469 api.ifindex_num = 0;
470
471 zapi_ipv4_add (zclient, p, &api);
472 }
473}
474
475void
476ospf_zebra_delete_discard (struct prefix_ipv4 *p)
477{
478 struct zapi_ipv4 api;
479
480 if (zclient->redist[ZEBRA_ROUTE_OSPF])
481 {
482 api.type = ZEBRA_ROUTE_OSPF;
483 api.flags = ZEBRA_FLAG_BLACKHOLE;
484 api.message = 0;
485 SET_FLAG (api.message, ZAPI_MESSAGE_NEXTHOP);
486 api.nexthop_num = 0;
487 api.ifindex_num = 0;
488
489 zapi_ipv4_delete (zclient, p, &api);
paul72357f22003-06-19 02:11:23 +0000490
491 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
paulcf795c52003-06-19 02:13:25 +0000492 zlog_info ("Zebra: Route delete discard %s/%d",
493 inet_ntoa (p->prefix), p->prefixlen);
paul72357f22003-06-19 02:11:23 +0000494
paul718e3742002-12-13 20:15:29 +0000495 }
496}
497
498int
499ospf_is_type_redistributed (int type)
500{
501 return (DEFAULT_ROUTE_TYPE (type)) ?
502 zclient->default_information : zclient->redist[type];
503}
504
505int
paul020709f2003-04-04 02:44:16 +0000506ospf_redistribute_set (struct ospf *ospf, int type, int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000507{
508 int force = 0;
paulcf795c52003-06-19 02:13:25 +0000509
paul718e3742002-12-13 20:15:29 +0000510 if (ospf_is_type_redistributed (type))
511 {
paul68980082003-03-25 05:07:42 +0000512 if (mtype != ospf->dmetric[type].type)
paulcf795c52003-06-19 02:13:25 +0000513 {
514 ospf->dmetric[type].type = mtype;
515 force = LSA_REFRESH_FORCE;
516 }
paul68980082003-03-25 05:07:42 +0000517 if (mvalue != ospf->dmetric[type].value)
paulcf795c52003-06-19 02:13:25 +0000518 {
519 ospf->dmetric[type].value = mvalue;
520 force = LSA_REFRESH_FORCE;
521 }
522
paul68980082003-03-25 05:07:42 +0000523 ospf_external_lsa_refresh_type (ospf, type, force);
paulcf795c52003-06-19 02:13:25 +0000524
paul718e3742002-12-13 20:15:29 +0000525 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
paulcf795c52003-06-19 02:13:25 +0000526 zlog_info ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
527 LOOKUP (ospf_redistributed_proto, type),
528 metric_type (ospf, type), metric_value (ospf, type));
529
paul718e3742002-12-13 20:15:29 +0000530 return CMD_SUCCESS;
531 }
532
paul68980082003-03-25 05:07:42 +0000533 ospf->dmetric[type].type = mtype;
534 ospf->dmetric[type].value = mvalue;
paul718e3742002-12-13 20:15:29 +0000535
536 zclient_redistribute_set (zclient, type);
537
538 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
539 zlog_info ("Redistribute[%s]: Start Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000540 LOOKUP (ospf_redistributed_proto, type),
541 metric_type (ospf, type), metric_value (ospf, type));
542
paul68980082003-03-25 05:07:42 +0000543 ospf_asbr_status_update (ospf, ++ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000544
545 return CMD_SUCCESS;
546}
547
548int
paul020709f2003-04-04 02:44:16 +0000549ospf_redistribute_unset (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000550{
551 if (type == zclient->redist_default)
552 return CMD_SUCCESS;
553
paulcf795c52003-06-19 02:13:25 +0000554 if (!ospf_is_type_redistributed (type))
paul718e3742002-12-13 20:15:29 +0000555 return CMD_SUCCESS;
556
557 zclient_redistribute_unset (zclient, type);
paulcf795c52003-06-19 02:13:25 +0000558
paul718e3742002-12-13 20:15:29 +0000559 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
560 zlog_info ("Redistribute[%s]: Stop",
paulcf795c52003-06-19 02:13:25 +0000561 LOOKUP (ospf_redistributed_proto, type));
paul718e3742002-12-13 20:15:29 +0000562
paul68980082003-03-25 05:07:42 +0000563 ospf->dmetric[type].type = -1;
564 ospf->dmetric[type].value = -1;
paul718e3742002-12-13 20:15:29 +0000565
566 /* Remove the routes from OSPF table. */
567 ospf_redistribute_withdraw (type);
568
paul68980082003-03-25 05:07:42 +0000569 ospf_asbr_status_update (ospf, --ospf->redistribute);
paul718e3742002-12-13 20:15:29 +0000570
571 return CMD_SUCCESS;
572}
573
574int
paul020709f2003-04-04 02:44:16 +0000575ospf_redistribute_default_set (struct ospf *ospf, int originate,
paulcf795c52003-06-19 02:13:25 +0000576 int mtype, int mvalue)
paul718e3742002-12-13 20:15:29 +0000577{
578 int force = 0;
paul020709f2003-04-04 02:44:16 +0000579
paul718e3742002-12-13 20:15:29 +0000580 if (ospf_is_type_redistributed (DEFAULT_ROUTE))
581 {
paul68980082003-03-25 05:07:42 +0000582 if (mtype != ospf->dmetric[DEFAULT_ROUTE].type)
paulcf795c52003-06-19 02:13:25 +0000583 {
584 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
585 force = 1;
586 }
paul68980082003-03-25 05:07:42 +0000587 if (mvalue != ospf->dmetric[DEFAULT_ROUTE].value)
paulcf795c52003-06-19 02:13:25 +0000588 {
589 force = 1;
590 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
591 }
592
paul68980082003-03-25 05:07:42 +0000593 ospf_external_lsa_refresh_default (ospf);
paulcf795c52003-06-19 02:13:25 +0000594
paul718e3742002-12-13 20:15:29 +0000595 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
paulcf795c52003-06-19 02:13:25 +0000596 zlog_info ("Redistribute[%s]: Refresh Type[%d], Metric[%d]",
597 LOOKUP (ospf_redistributed_proto, DEFAULT_ROUTE),
598 metric_type (ospf, DEFAULT_ROUTE),
599 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000600 return CMD_SUCCESS;
601 }
602
paul68980082003-03-25 05:07:42 +0000603 ospf->default_originate = originate;
604 ospf->dmetric[DEFAULT_ROUTE].type = mtype;
605 ospf->dmetric[DEFAULT_ROUTE].value = mvalue;
paul718e3742002-12-13 20:15:29 +0000606
607 zclient_redistribute_default_set (zclient);
paulcf795c52003-06-19 02:13:25 +0000608
paul718e3742002-12-13 20:15:29 +0000609 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
610 zlog_info ("Redistribute[DEFAULT]: Start Type[%d], Metric[%d]",
paulcf795c52003-06-19 02:13:25 +0000611 metric_type (ospf, DEFAULT_ROUTE),
612 metric_value (ospf, DEFAULT_ROUTE));
paul718e3742002-12-13 20:15:29 +0000613
paul68980082003-03-25 05:07:42 +0000614 if (ospf->router_id.s_addr == 0)
615 ospf->external_origin |= (1 << DEFAULT_ROUTE);
paul718e3742002-12-13 20:15:29 +0000616 else
617 thread_add_timer (master, ospf_default_originate_timer,
paulcf795c52003-06-19 02:13:25 +0000618 &ospf->default_originate, 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
635 zclient_redistribute_default_unset (zclient);
636
637 if (IS_DEBUG_OSPF (zebra, ZEBRA_REDISTRIBUTE))
638 zlog_info ("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
645int
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))
712 zlog_info ("Redistribute[%s]: %s/%d filtered by ditribute-list.",
713 LOOKUP (ospf_redistributed_proto, type),
714 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))
733 zlog_info ("Redistribute[%s]: %s/%d filtered by route-map.",
734 LOOKUP (ospf_redistributed_proto, type),
735 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
paul020709f2003-04-04 02:44:16 +0000750ospf_routemap_set (struct ospf *ospf, int type, 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. */
770int
771ospf_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:
824 * ignore reject/blackhole routes
825 * need a better generalised solution for these types
826 * really.
827 */
828 if ( CHECK_FLAG (api.flags, ZEBRA_FLAG_BLACKHOLE)
829 || CHECK_FLAG (api.flags, ZEBRA_FLAG_REJECT))
830 return 0;
831
paul718e3742002-12-13 20:15:29 +0000832 ei = ospf_external_info_add (api.type, p, ifindex, nexthop);
833
paul68980082003-03-25 05:07:42 +0000834 if (ospf->router_id.s_addr == 0)
paulcf795c52003-06-19 02:13:25 +0000835 /* Set flags to generate AS-external-LSA originate event
836 for each redistributed protocols later. */
837 ospf->external_origin |= (1 << api.type);
paul718e3742002-12-13 20:15:29 +0000838 else
paulcf795c52003-06-19 02:13:25 +0000839 {
840 if (ei)
841 {
842 if (is_prefix_default (&p))
843 ospf_external_lsa_refresh_default (ospf);
844 else
845 {
846 struct ospf_lsa *current;
paul718e3742002-12-13 20:15:29 +0000847
paulcf795c52003-06-19 02:13:25 +0000848 current = ospf_external_info_find_lsa (ospf, &ei->p);
849 if (!current)
850 ospf_external_lsa_originate (ospf, ei);
851 else if (IS_LSA_MAXAGE (current))
852 ospf_external_lsa_refresh (ospf, current,
853 ei, LSA_REFRESH_FORCE);
854 else
855 zlog_warn ("ospf_zebra_read_ipv4() : %s already exists",
856 inet_ntoa (p.prefix));
857 }
858 }
859 }
paul718e3742002-12-13 20:15:29 +0000860 }
paulcf795c52003-06-19 02:13:25 +0000861 else /* if (command == ZEBRA_IPV4_ROUTE_DELETE) */
paul718e3742002-12-13 20:15:29 +0000862 {
863 ospf_external_info_delete (api.type, p);
paul7021c422003-07-15 12:52:22 +0000864 if (is_prefix_default (&p))
paulcf795c52003-06-19 02:13:25 +0000865 ospf_external_lsa_refresh_default (ospf);
paul7021c422003-07-15 12:52:22 +0000866 else
867 ospf_external_lsa_flush (ospf, api.type, &p, ifindex, nexthop);
paul718e3742002-12-13 20:15:29 +0000868 }
869
870 return 0;
871}
paul718e3742002-12-13 20:15:29 +0000872
paulcf795c52003-06-19 02:13:25 +0000873
paul718e3742002-12-13 20:15:29 +0000874int
paul68980082003-03-25 05:07:42 +0000875ospf_distribute_list_out_set (struct ospf *ospf, int type, char *name)
paul718e3742002-12-13 20:15:29 +0000876{
877 /* Lookup access-list for distribute-list. */
paul020709f2003-04-04 02:44:16 +0000878 DISTRIBUTE_LIST (ospf, type) = access_list_lookup (AFI_IP, name);
paul718e3742002-12-13 20:15:29 +0000879
880 /* Clear previous distribute-name. */
paul020709f2003-04-04 02:44:16 +0000881 if (DISTRIBUTE_NAME (ospf, type))
882 free (DISTRIBUTE_NAME (ospf, type));
paul718e3742002-12-13 20:15:29 +0000883
884 /* Set distribute-name. */
paul020709f2003-04-04 02:44:16 +0000885 DISTRIBUTE_NAME (ospf, type) = strdup (name);
paul718e3742002-12-13 20:15:29 +0000886
887 /* If access-list have been set, schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000888 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000889 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000890
891 return CMD_SUCCESS;
892}
893
894int
paul68980082003-03-25 05:07:42 +0000895ospf_distribute_list_out_unset (struct ospf *ospf, int type, char *name)
paul718e3742002-12-13 20:15:29 +0000896{
897 /* Schedule update timer. */
paul020709f2003-04-04 02:44:16 +0000898 if (DISTRIBUTE_LIST (ospf, type))
paul68980082003-03-25 05:07:42 +0000899 ospf_distribute_list_update (ospf, type);
paul718e3742002-12-13 20:15:29 +0000900
901 /* Unset distribute-list. */
paul020709f2003-04-04 02:44:16 +0000902 DISTRIBUTE_LIST (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000903
904 /* Clear distribute-name. */
paul020709f2003-04-04 02:44:16 +0000905 if (DISTRIBUTE_NAME (ospf, type))
906 free (DISTRIBUTE_NAME (ospf, type));
paulcf795c52003-06-19 02:13:25 +0000907
paul020709f2003-04-04 02:44:16 +0000908 DISTRIBUTE_NAME (ospf, type) = NULL;
paul718e3742002-12-13 20:15:29 +0000909
910 return CMD_SUCCESS;
911}
912
913/* distribute-list update timer. */
914int
915ospf_distribute_list_update_timer (struct thread *thread)
916{
917 struct route_node *rn;
918 struct external_info *ei;
919 struct route_table *rt;
920 struct ospf_lsa *lsa;
921 u_char type;
paul020709f2003-04-04 02:44:16 +0000922 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000923
924 type = (int) THREAD_ARG (thread);
925 rt = EXTERNAL_INFO (type);
926
paul020709f2003-04-04 02:44:16 +0000927 ospf = ospf_lookup ();
928 if (ospf == NULL)
929 return 0;
930
paul68980082003-03-25 05:07:42 +0000931 ospf->t_distribute_update = NULL;
paul718e3742002-12-13 20:15:29 +0000932
933 zlog_info ("Zebra[Redistribute]: distribute-list update timer fired!");
934
935 /* foreach all external info. */
936 if (rt)
937 for (rn = route_top (rt); rn; rn = route_next (rn))
938 if ((ei = rn->info) != NULL)
paulcf795c52003-06-19 02:13:25 +0000939 {
940 if (is_prefix_default (&ei->p))
941 ospf_external_lsa_refresh_default (ospf);
942 else if ((lsa = ospf_external_info_find_lsa (ospf, &ei->p)))
943 ospf_external_lsa_refresh (ospf, lsa, ei, LSA_REFRESH_IF_CHANGED);
944 else
945 ospf_external_lsa_originate (ospf, ei);
946 }
paul718e3742002-12-13 20:15:29 +0000947 return 0;
948}
949
950#define OSPF_DISTRIBUTE_UPDATE_DELAY 5
951
952/* Update distribute-list and set timer to apply access-list. */
953void
paul68980082003-03-25 05:07:42 +0000954ospf_distribute_list_update (struct ospf *ospf, int type)
paul718e3742002-12-13 20:15:29 +0000955{
956 struct route_table *rt;
paulcf795c52003-06-19 02:13:25 +0000957
paul718e3742002-12-13 20:15:29 +0000958 /* External info does not exist. */
959 if (!(rt = EXTERNAL_INFO (type)))
960 return;
961
962 /* If exists previously invoked thread, then cancel it. */
paul68980082003-03-25 05:07:42 +0000963 if (ospf->t_distribute_update)
964 OSPF_TIMER_OFF (ospf->t_distribute_update);
paul718e3742002-12-13 20:15:29 +0000965
966 /* Set timer. */
paul68980082003-03-25 05:07:42 +0000967 ospf->t_distribute_update =
paul718e3742002-12-13 20:15:29 +0000968 thread_add_timer (master, ospf_distribute_list_update_timer,
paulcf795c52003-06-19 02:13:25 +0000969 (void *) type, OSPF_DISTRIBUTE_UPDATE_DELAY);
paul718e3742002-12-13 20:15:29 +0000970}
971
972/* If access-list is updated, apply some check. */
973void
974ospf_filter_update (struct access_list *access)
975{
paul020709f2003-04-04 02:44:16 +0000976 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +0000977 int type;
978 int abr_inv = 0;
979 struct ospf_area *area;
980 listnode node;
981
982 /* If OSPF instatnce does not exist, return right now. */
paul020709f2003-04-04 02:44:16 +0000983 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +0000984 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +0000985 return;
986
paul718e3742002-12-13 20:15:29 +0000987 /* Update distribute-list, and apply filter. */
988 for (type = 0; type < ZEBRA_ROUTE_MAX; type++)
989 {
paul020709f2003-04-04 02:44:16 +0000990 if (ROUTEMAP (ospf, type) != NULL)
paulcf795c52003-06-19 02:13:25 +0000991 {
992 /* if route-map is not NULL it may be using this access list */
993 ospf_distribute_list_update (ospf, type);
994 continue;
995 }
996
paul718e3742002-12-13 20:15:29 +0000997
paul020709f2003-04-04 02:44:16 +0000998 if (DISTRIBUTE_NAME (ospf, type))
paulcf795c52003-06-19 02:13:25 +0000999 {
1000 /* Keep old access-list for distribute-list. */
1001 struct access_list *old = DISTRIBUTE_LIST (ospf, type);
1002
1003 /* Update access-list for distribute-list. */
1004 DISTRIBUTE_LIST (ospf, type) =
1005 access_list_lookup (AFI_IP, DISTRIBUTE_NAME (ospf, type));
1006
1007 /* No update for this distribute type. */
1008 if (old == NULL && DISTRIBUTE_LIST (ospf, type) == NULL)
1009 continue;
1010
1011 /* Schedule distribute-list update timer. */
1012 if (DISTRIBUTE_LIST (ospf, type) == NULL ||
1013 strcmp (DISTRIBUTE_NAME (ospf, type), access->name) == 0)
1014 ospf_distribute_list_update (ospf, type);
1015 }
paul718e3742002-12-13 20:15:29 +00001016 }
1017
1018 /* Update Area access-list. */
paul68980082003-03-25 05:07:42 +00001019 for (node = listhead (ospf->areas); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +00001020 if ((area = getdata (node)) != NULL)
1021 {
paulcf795c52003-06-19 02:13:25 +00001022 if (EXPORT_NAME (area))
1023 {
1024 EXPORT_LIST (area) = NULL;
1025 abr_inv++;
1026 }
paul718e3742002-12-13 20:15:29 +00001027
paulcf795c52003-06-19 02:13:25 +00001028 if (IMPORT_NAME (area))
1029 {
1030 IMPORT_LIST (area) = NULL;
1031 abr_inv++;
1032 }
paul718e3742002-12-13 20:15:29 +00001033 }
1034
1035 /* Schedule ABR tasks -- this will be changed -- takada. */
paul020709f2003-04-04 02:44:16 +00001036 if (IS_OSPF_ABR (ospf) && abr_inv)
paul68980082003-03-25 05:07:42 +00001037 ospf_schedule_abr_task (ospf);
paul718e3742002-12-13 20:15:29 +00001038}
paul718e3742002-12-13 20:15:29 +00001039
paulcf795c52003-06-19 02:13:25 +00001040
paul718e3742002-12-13 20:15:29 +00001041struct ospf_distance *
1042ospf_distance_new ()
1043{
1044 struct ospf_distance *new;
1045 new = XMALLOC (MTYPE_OSPF_DISTANCE, sizeof (struct ospf_distance));
1046 memset (new, 0, sizeof (struct ospf_distance));
1047 return new;
1048}
1049
1050void
1051ospf_distance_free (struct ospf_distance *odistance)
1052{
1053 XFREE (MTYPE_OSPF_DISTANCE, odistance);
1054}
1055
1056int
paul020709f2003-04-04 02:44:16 +00001057ospf_distance_set (struct vty *vty, struct ospf *ospf, char *distance_str,
paulcf795c52003-06-19 02:13:25 +00001058 char *ip_str, char *access_list_str)
paul718e3742002-12-13 20:15:29 +00001059{
1060 int ret;
1061 struct prefix_ipv4 p;
1062 u_char distance;
1063 struct route_node *rn;
1064 struct ospf_distance *odistance;
1065
1066 ret = str2prefix_ipv4 (ip_str, &p);
1067 if (ret == 0)
1068 {
1069 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1070 return CMD_WARNING;
1071 }
1072
1073 distance = atoi (distance_str);
1074
1075 /* Get OSPF distance node. */
paul68980082003-03-25 05:07:42 +00001076 rn = route_node_get (ospf->distance_table, (struct prefix *) &p);
paul718e3742002-12-13 20:15:29 +00001077 if (rn->info)
1078 {
1079 odistance = rn->info;
1080 route_unlock_node (rn);
1081 }
1082 else
1083 {
1084 odistance = ospf_distance_new ();
1085 rn->info = odistance;
1086 }
1087
1088 /* Set distance value. */
1089 odistance->distance = distance;
1090
1091 /* Reset access-list configuration. */
1092 if (odistance->access_list)
1093 {
1094 free (odistance->access_list);
1095 odistance->access_list = NULL;
1096 }
1097 if (access_list_str)
1098 odistance->access_list = strdup (access_list_str);
1099
1100 return CMD_SUCCESS;
1101}
1102
1103int
paul020709f2003-04-04 02:44:16 +00001104ospf_distance_unset (struct vty *vty, struct ospf *ospf, char *distance_str,
paulcf795c52003-06-19 02:13:25 +00001105 char *ip_str, char *access_list_str)
paul718e3742002-12-13 20:15:29 +00001106{
1107 int ret;
1108 struct prefix_ipv4 p;
1109 u_char distance;
1110 struct route_node *rn;
1111 struct ospf_distance *odistance;
1112
1113 ret = str2prefix_ipv4 (ip_str, &p);
1114 if (ret == 0)
1115 {
1116 vty_out (vty, "Malformed prefix%s", VTY_NEWLINE);
1117 return CMD_WARNING;
1118 }
1119
1120 distance = atoi (distance_str);
1121
paulcf795c52003-06-19 02:13:25 +00001122 rn = route_node_lookup (ospf->distance_table, (struct prefix *) &p);
1123 if (!rn)
paul718e3742002-12-13 20:15:29 +00001124 {
1125 vty_out (vty, "Can't find specified prefix%s", VTY_NEWLINE);
1126 return CMD_WARNING;
1127 }
1128
1129 odistance = rn->info;
1130
1131 if (odistance->access_list)
1132 free (odistance->access_list);
1133 ospf_distance_free (odistance);
1134
1135 rn->info = NULL;
1136 route_unlock_node (rn);
1137 route_unlock_node (rn);
1138
1139 return CMD_SUCCESS;
1140}
1141
1142void
paul68980082003-03-25 05:07:42 +00001143ospf_distance_reset (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +00001144{
1145 struct route_node *rn;
1146 struct ospf_distance *odistance;
1147
paul68980082003-03-25 05:07:42 +00001148 for (rn = route_top (ospf->distance_table); rn; rn = route_next (rn))
paul718e3742002-12-13 20:15:29 +00001149 if ((odistance = rn->info) != NULL)
1150 {
paulcf795c52003-06-19 02:13:25 +00001151 if (odistance->access_list)
1152 free (odistance->access_list);
1153 ospf_distance_free (odistance);
1154 rn->info = NULL;
1155 route_unlock_node (rn);
paul718e3742002-12-13 20:15:29 +00001156 }
1157}
1158
1159u_char
1160ospf_distance_apply (struct prefix_ipv4 *p, struct ospf_route *or)
1161{
paul020709f2003-04-04 02:44:16 +00001162 struct ospf *ospf;
paul718e3742002-12-13 20:15:29 +00001163
paul020709f2003-04-04 02:44:16 +00001164 ospf = ospf_lookup ();
paul68980082003-03-25 05:07:42 +00001165 if (ospf == NULL)
paul718e3742002-12-13 20:15:29 +00001166 return 0;
1167
paul68980082003-03-25 05:07:42 +00001168 if (ospf->distance_intra)
paul718e3742002-12-13 20:15:29 +00001169 if (or->path_type == OSPF_PATH_INTRA_AREA)
paul68980082003-03-25 05:07:42 +00001170 return ospf->distance_intra;
paul718e3742002-12-13 20:15:29 +00001171
paul68980082003-03-25 05:07:42 +00001172 if (ospf->distance_inter)
paul718e3742002-12-13 20:15:29 +00001173 if (or->path_type == OSPF_PATH_INTER_AREA)
paul68980082003-03-25 05:07:42 +00001174 return ospf->distance_inter;
paul718e3742002-12-13 20:15:29 +00001175
paul68980082003-03-25 05:07:42 +00001176 if (ospf->distance_external)
paul718e3742002-12-13 20:15:29 +00001177 if (or->path_type == OSPF_PATH_TYPE1_EXTERNAL
paulcf795c52003-06-19 02:13:25 +00001178 || or->path_type == OSPF_PATH_TYPE2_EXTERNAL)
paul68980082003-03-25 05:07:42 +00001179 return ospf->distance_external;
paulcf795c52003-06-19 02:13:25 +00001180
paul68980082003-03-25 05:07:42 +00001181 if (ospf->distance_all)
1182 return ospf->distance_all;
paul718e3742002-12-13 20:15:29 +00001183
1184 return 0;
1185}
1186
1187void
1188ospf_zebra_init ()
1189{
1190 /* Allocate zebra structure. */
1191 zclient = zclient_new ();
1192 zclient_init (zclient, ZEBRA_ROUTE_OSPF);
1193 zclient->interface_add = ospf_interface_add;
1194 zclient->interface_delete = ospf_interface_delete;
1195 zclient->interface_up = ospf_interface_state_up;
1196 zclient->interface_down = ospf_interface_state_down;
1197 zclient->interface_address_add = ospf_interface_address_add;
1198 zclient->interface_address_delete = ospf_interface_address_delete;
1199 zclient->ipv4_route_add = ospf_zebra_read_ipv4;
1200 zclient->ipv4_route_delete = ospf_zebra_read_ipv4;
1201
1202 access_list_add_hook (ospf_filter_update);
1203 access_list_delete_hook (ospf_filter_update);
1204}