blob: 83abf5538fd00f5929249f5d5f93a41456cf428e [file] [log] [blame]
paul718e3742002-12-13 20:15:29 +00001/*
2 * OSPF Interface functions.
3 * Copyright (C) 1999, 2000 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
8 * it under the terms of the GNU General Public License as published
9 * by the Free Software Foundation; either version 2, or (at your
10 * option) any 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 "linklist.h"
27#include "prefix.h"
28#include "if.h"
29#include "table.h"
30#include "memory.h"
31#include "command.h"
32#include "stream.h"
33#include "log.h"
34
paul68980082003-03-25 05:07:42 +000035#include "ospfd/ospfd.h"
paul718e3742002-12-13 20:15:29 +000036#include "ospfd/ospf_spf.h"
37#include "ospfd/ospf_interface.h"
38#include "ospfd/ospf_ism.h"
39#include "ospfd/ospf_asbr.h"
40#include "ospfd/ospf_lsa.h"
41#include "ospfd/ospf_lsdb.h"
42#include "ospfd/ospf_neighbor.h"
43#include "ospfd/ospf_nsm.h"
44#include "ospfd/ospf_packet.h"
45#include "ospfd/ospf_abr.h"
paul718e3742002-12-13 20:15:29 +000046#include "ospfd/ospf_network.h"
47#include "ospfd/ospf_dump.h"
48#ifdef HAVE_SNMP
49#include "ospfd/ospf_snmp.h"
50#endif /* HAVE_SNMP */
51
52
53int
54ospf_if_get_output_cost (struct ospf_interface *oi)
55{
56 /* If all else fails, use default OSPF cost */
57 u_int32_t cost;
58 u_int32_t bw, refbw;
59
60 bw = oi->ifp->bandwidth ? oi->ifp->bandwidth : OSPF_DEFAULT_BANDWIDTH;
paul68980082003-03-25 05:07:42 +000061 refbw = oi->ospf->ref_bandwidth;
paul718e3742002-12-13 20:15:29 +000062
63 /* A specifed ip ospf cost overrides a calculated one. */
64 if (OSPF_IF_PARAM_CONFIGURED (IF_DEF_PARAMS (oi->ifp), output_cost_cmd) ||
65 OSPF_IF_PARAM_CONFIGURED (oi->params, output_cost_cmd))
66 cost = OSPF_IF_PARAM (oi, output_cost_cmd);
67 /* See if a cost can be calculated from the zebra processes
68 interface bandwidth field. */
69 else
70 {
71 cost = (u_int32_t) ((double)refbw / (double)bw + (double)0.5);
72 if (cost < 1)
73 cost = 1;
74 else if (cost > 65535)
75 cost = 65535;
76 }
77
78 return cost;
79}
80
81void
82ospf_if_recalculate_output_cost (struct interface *ifp)
83{
84 u_int32_t newcost;
85 struct route_node *rn;
86
87 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
88 {
89 struct ospf_interface *oi;
90
91 if ( (oi = rn->info) == NULL)
92 continue;
93
94 newcost = ospf_if_get_output_cost (oi);
95
96 /* Is actual output cost changed? */
97 if (oi->output_cost != newcost)
98 {
99 oi->output_cost = newcost;
100 ospf_router_lsa_timer_add (oi->area);
101 }
102 }
103}
104
105void
106ospf_if_reset_variables (struct ospf_interface *oi)
107{
108 /* Set default values. */
109 /* don't clear this flag. oi->flag = OSPF_IF_DISABLE; */
110
111 if (oi->vl_data)
112 oi->type = OSPF_IFTYPE_VIRTUALLINK;
113 else
114 /* preserve network-type */
115 if (oi->type != OSPF_IFTYPE_NBMA)
116 oi->type = OSPF_IFTYPE_BROADCAST;
117
118 oi->state = ISM_Down;
119
120 oi->crypt_seqnum = 0;
121
122 /* This must be short, (less than RxmtInterval)
123 - RFC 2328 Section 13.5 para 3. Set to 1 second to avoid Acks being
124 held back for too long - MAG */
125 oi->v_ls_ack = 1;
126}
127
128void
129ospf_add_to_if (struct interface *ifp, struct ospf_interface *oi)
130{
131 struct route_node *rn;
132 struct prefix p;
133
134 p = *oi->address;
135 p.prefixlen = IPV4_MAX_PREFIXLEN;
136
137 rn = route_node_get (IF_OIFS (ifp), &p);
paul8c80cb72003-02-18 23:25:44 +0000138 /* rn->info should either be NULL or equal to this oi
139 * as route_node_get may return an existing node
140 */
141 assert (! rn->info || rn->info == oi);
paul718e3742002-12-13 20:15:29 +0000142 rn->info = oi;
143}
144
145void
146ospf_delete_from_if (struct interface *ifp, struct ospf_interface *oi)
147{
148 struct route_node *rn;
149 struct prefix p;
150
151 p = *oi->address;
152 p.prefixlen = IPV4_MAX_PREFIXLEN;
153
154 rn = route_node_lookup (IF_OIFS (oi->ifp), &p);
155 assert (rn);
156 assert (rn->info);
157 rn->info = NULL;
158 route_unlock_node (rn);
159 route_unlock_node (rn);
160}
161
162struct ospf_interface *
paul68980082003-03-25 05:07:42 +0000163ospf_if_new (struct ospf *ospf, struct interface *ifp, struct prefix *p)
paul718e3742002-12-13 20:15:29 +0000164{
165 struct ospf_interface *oi;
166
167 oi = XCALLOC (MTYPE_OSPF_IF, sizeof (struct ospf_interface));
168 memset (oi, 0, sizeof (struct ospf_interface));
169
170 /* Set zebra interface pointer. */
171 oi->ifp = ifp;
172 oi->address = p;
173
174 ospf_add_to_if (ifp, oi);
paul68980082003-03-25 05:07:42 +0000175 listnode_add (ospf->oiflist, oi);
paul718e3742002-12-13 20:15:29 +0000176
177 /* Clear self-originated network-LSA. */
178 oi->network_lsa_self = NULL;
179
180 /* Initialize neighbor list. */
181 oi->nbrs = route_table_init ();
182
183 /* Initialize static neighbor list. */
184 oi->nbr_nbma = list_new ();
185
186 /* Initialize Link State Acknowledgment list. */
187 oi->ls_ack = list_new ();
188 oi->ls_ack_direct.ls_ack = list_new ();
189
190 /* Set default values. */
191 ospf_if_reset_variables (oi);
192
193 /* Add pseudo neighbor. */
194 oi->nbr_self = ospf_nbr_new (oi);
195 oi->nbr_self->state = NSM_TwoWay;
paul718e3742002-12-13 20:15:29 +0000196 oi->nbr_self->priority = OSPF_IF_PARAM (oi, priority);
197 oi->nbr_self->options = OSPF_OPTION_E;
198
199 oi->ls_upd_queue = route_table_init ();
200 oi->t_ls_upd_event = NULL;
201 oi->t_ls_ack_direct = NULL;
202
paul68980082003-03-25 05:07:42 +0000203 oi->crypt_seqnum = time (NULL);
204
paul718e3742002-12-13 20:15:29 +0000205#ifdef HAVE_OPAQUE_LSA
206 ospf_opaque_type9_lsa_init (oi);
207#endif /* HAVE_OPAQUE_LSA */
208
paul68980082003-03-25 05:07:42 +0000209 oi->ospf = ospf;
paul718e3742002-12-13 20:15:29 +0000210
211 return oi;
212}
213
214/* Restore an interface to its pre UP state
215 Used from ism_interface_down only */
216void
217ospf_if_cleanup (struct ospf_interface *oi)
218{
219 struct route_node *rn;
220 listnode node;
221 struct ospf_neighbor *nbr;
222
223 /* oi->nbrs and oi->nbr_nbma should be deletete on InterafceDown event */
224 /* delete all static neighbors attached to this interface */
225 for (node = listhead (oi->nbr_nbma); node; )
226 {
227 struct ospf_nbr_nbma *nbr_nbma = getdata (node);
228 nextnode (node);
229
230 OSPF_POLL_TIMER_OFF (nbr_nbma->t_poll);
231
232 if (nbr_nbma->nbr)
233 {
234 nbr_nbma->nbr->nbr_nbma = NULL;
235 nbr_nbma->nbr = NULL;
236 }
237
238 nbr_nbma->oi = NULL;
239
240 listnode_delete (oi->nbr_nbma, nbr_nbma);
241 }
242
243 /* send Neighbor event KillNbr to all associated neighbors. */
244 for (rn = route_top (oi->nbrs); rn; rn = route_next (rn))
245 if ((nbr = rn->info) != NULL)
246 if (nbr != oi->nbr_self)
247 OSPF_NSM_EVENT_EXECUTE (nbr, NSM_KillNbr);
248
249 /* Cleanup Link State Acknowlegdment list. */
250 for (node = listhead (oi->ls_ack); node; nextnode (node))
251 ospf_lsa_unlock (node->data);
252 list_delete_all_node (oi->ls_ack);
253
254 oi->crypt_seqnum = 0;
255
256 /* Empty link state update queue */
257 ospf_ls_upd_queue_empty (oi);
258
259 /* Handle pseudo neighbor. */
260 ospf_nbr_delete (oi->nbr_self);
261 oi->nbr_self = ospf_nbr_new (oi);
262 oi->nbr_self->state = NSM_TwoWay;
263 oi->nbr_self->priority = OSPF_IF_PARAM (oi, priority);
paulf2c80652002-12-13 21:44:27 +0000264
265 switch (oi->area->external_routing)
266 {
267 case OSPF_AREA_DEFAULT:
268 SET_FLAG (oi->nbr_self->options, OSPF_OPTION_E);
269 break;
270 case OSPF_AREA_STUB:
271 UNSET_FLAG (oi->nbr_self->options, OSPF_OPTION_E);
272 break;
273#ifdef HAVE_NSSA
274 case OSPF_AREA_NSSA:
275 UNSET_FLAG (oi->nbr_self->options, OSPF_OPTION_E);
276 SET_FLAG (oi->nbr_self->options, OSPF_OPTION_NP);
277 break;
278#endif /* HAVE_NSSA */
279 }
paul718e3742002-12-13 20:15:29 +0000280
281 ospf_lsa_unlock (oi->network_lsa_self);
282 oi->network_lsa_self = NULL;
283 OSPF_TIMER_OFF (oi->t_network_lsa_self);
284}
285
286void
287ospf_if_free (struct ospf_interface *oi)
288{
289 ospf_if_down (oi);
290
291 assert (oi->state == ISM_Down);
292
293#ifdef HAVE_OPAQUE_LSA
294 ospf_opaque_type9_lsa_term (oi);
295#endif /* HAVE_OPAQUE_LSA */
296
297 /* Free Pseudo Neighbour */
298 ospf_nbr_delete (oi->nbr_self);
299
300 route_table_finish (oi->nbrs);
301 route_table_finish (oi->ls_upd_queue);
302
303 /* Free any lists that should be freed */
304 list_free (oi->nbr_nbma);
305
306 list_free (oi->ls_ack);
307 list_free (oi->ls_ack_direct.ls_ack);
308
309 ospf_delete_from_if (oi->ifp, oi);
310
paul68980082003-03-25 05:07:42 +0000311 listnode_delete (oi->ospf->oiflist, oi);
paul718e3742002-12-13 20:15:29 +0000312 listnode_delete (oi->area->oiflist, oi);
313
314 memset (oi, 0, sizeof (*oi));
315 XFREE (MTYPE_OSPF_IF, oi);
316}
317
318
319/*
320* check if interface with given address is configured and
321* return it if yes.
322*/
323struct ospf_interface *
paul68980082003-03-25 05:07:42 +0000324ospf_if_is_configured (struct ospf *ospf, struct in_addr *address)
paul718e3742002-12-13 20:15:29 +0000325{
326 listnode node;
327 struct ospf_interface *oi;
328 struct prefix *addr;
329
paul68980082003-03-25 05:07:42 +0000330 for (node = listhead (ospf->oiflist); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +0000331 if ((oi = getdata (node)) != NULL && oi->type != OSPF_IFTYPE_VIRTUALLINK)
332 {
333 if (oi->type == OSPF_IFTYPE_POINTOPOINT)
334 addr = oi->connected->destination;
335 else
336 addr = oi->address;
337
338 if (IPV4_ADDR_SAME (address, &addr->u.prefix4))
339 return oi;
340 }
341
342 return NULL;
343}
344
345int
346ospf_if_is_up (struct ospf_interface *oi)
347{
348 return if_is_up (oi->ifp);
349}
350
351struct ospf_interface *
paul68980082003-03-25 05:07:42 +0000352ospf_if_lookup_by_local_addr (struct ospf *ospf,
353 struct interface *ifp, struct in_addr address)
paul718e3742002-12-13 20:15:29 +0000354{
355 listnode node;
356 struct ospf_interface *oi;
357
paul68980082003-03-25 05:07:42 +0000358 for (node = listhead (ospf->oiflist); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +0000359 if ((oi = getdata (node)) != NULL && oi->type != OSPF_IFTYPE_VIRTUALLINK)
360 {
361 if (ifp && oi->ifp != ifp)
362 continue;
363
364 if (IPV4_ADDR_SAME (&address, &oi->address->u.prefix4))
365 return oi;
366 }
367
368 return NULL;
369}
370
371struct ospf_interface *
paul68980082003-03-25 05:07:42 +0000372ospf_if_lookup_by_prefix (struct ospf *ospf, struct prefix_ipv4 *p)
paul718e3742002-12-13 20:15:29 +0000373{
374 listnode node;
375 struct ospf_interface *oi;
376 struct prefix ptmp;
377
378 /* Check each Interface. */
paul68980082003-03-25 05:07:42 +0000379 for (node = listhead (ospf->oiflist); node; nextnode (node))
380 {
381 if ((oi = getdata (node)) != NULL && oi->type != OSPF_IFTYPE_VIRTUALLINK)
382 {
383 if (oi->type == OSPF_IFTYPE_POINTOPOINT)
384 {
385 prefix_copy (&ptmp, oi->connected->destination);
386 ptmp.prefixlen = IPV4_MAX_BITLEN;
387 }
388 else
389 prefix_copy (&ptmp, oi->address);
paul718e3742002-12-13 20:15:29 +0000390
paul68980082003-03-25 05:07:42 +0000391 apply_mask (&ptmp);
392 if (prefix_same (&ptmp, (struct prefix *) p))
393 return oi;
394 }
395 }
paul718e3742002-12-13 20:15:29 +0000396 return NULL;
397}
398
399/* determine receiving interface by source of packet */
400struct ospf_interface *
paul68980082003-03-25 05:07:42 +0000401ospf_if_lookup_recv_if (struct ospf *ospf, struct in_addr src)
paul718e3742002-12-13 20:15:29 +0000402{
403 listnode node;
404 struct prefix_ipv4 addr;
405 struct ospf_interface *oi, *match;
406
407 addr.family = AF_INET;
408 addr.prefix = src;
409 addr.prefixlen = IPV4_MAX_BITLEN;
410
411 match = NULL;
412
paul68980082003-03-25 05:07:42 +0000413 for (node = listhead (ospf->oiflist); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +0000414 {
415 oi = getdata (node);
416
417 if (oi->type == OSPF_IFTYPE_VIRTUALLINK)
418 continue;
419
420 if (oi->type == OSPF_IFTYPE_POINTOPOINT)
421 {
422 if (IPV4_ADDR_SAME (&oi->connected->destination->u.prefix4, &src))
423 return oi;
424 }
425 else
426 {
427 if (prefix_match (oi->address, (struct prefix *) &addr))
paul0a825c72003-05-24 13:48:16 +0000428 {
paulaf8d0332003-05-24 15:31:45 +0000429 if ( (match == NULL) ||
430 (match->address->prefixlen < oi->address->prefixlen)
431 )
paul0a825c72003-05-24 13:48:16 +0000432 match = oi;
433 }
paul718e3742002-12-13 20:15:29 +0000434 }
435 }
436
437 return match;
438}
439
440void
441ospf_if_stream_set (struct ospf_interface *oi)
442{
443 /* set output fifo queue. */
444 if (oi->obuf == NULL)
445 oi->obuf = ospf_fifo_new ();
446}
447
448void
449ospf_if_stream_unset (struct ospf_interface *oi)
450{
paul68980082003-03-25 05:07:42 +0000451 struct ospf *ospf = oi->ospf;
452
paul718e3742002-12-13 20:15:29 +0000453 if (oi->obuf)
454 {
455 ospf_fifo_free (oi->obuf);
456 oi->obuf = NULL;
457
458 if (oi->on_write_q)
459 {
paul68980082003-03-25 05:07:42 +0000460 listnode_delete (ospf->oi_write_q, oi);
461 if (list_isempty(ospf->oi_write_q))
462 OSPF_TIMER_OFF (ospf->t_write);
paul718e3742002-12-13 20:15:29 +0000463 oi->on_write_q = 0;
464 }
465 }
466}
paul68980082003-03-25 05:07:42 +0000467
paul718e3742002-12-13 20:15:29 +0000468
469struct ospf_if_params *
470ospf_new_if_params ()
471{
472 struct ospf_if_params *oip;
473
474 oip = XMALLOC (MTYPE_OSPF_IF_PARAMS, sizeof (struct ospf_if_params));
paul718e3742002-12-13 20:15:29 +0000475
476 if (!oip)
477 return NULL;
478
paulf6457892003-04-17 16:11:30 +0000479 memset (oip, 0, sizeof (struct ospf_if_params));
480
paul718e3742002-12-13 20:15:29 +0000481 UNSET_IF_PARAM (oip, output_cost_cmd);
482 UNSET_IF_PARAM (oip, transmit_delay);
483 UNSET_IF_PARAM (oip, retransmit_interval);
484 UNSET_IF_PARAM (oip, passive_interface);
485 UNSET_IF_PARAM (oip, v_hello);
486 UNSET_IF_PARAM (oip, v_wait);
487 UNSET_IF_PARAM (oip, priority);
488 UNSET_IF_PARAM (oip, type);
489 UNSET_IF_PARAM (oip, auth_simple);
490 UNSET_IF_PARAM (oip, auth_crypt);
491 UNSET_IF_PARAM (oip, auth_type);
paulec1ca632003-06-04 02:23:15 +0000492
493 oip->auth_crypt = list_new ();
paul718e3742002-12-13 20:15:29 +0000494
paul718e3742002-12-13 20:15:29 +0000495 return oip;
496}
497
498void
499ospf_del_if_params (struct ospf_if_params *oip)
500{
501 list_delete (oip->auth_crypt);
502 XFREE (MTYPE_OSPF_IF_PARAMS, oip);
503}
504
505void
506ospf_free_if_params (struct interface *ifp, struct in_addr addr)
507{
508 struct ospf_if_params *oip;
509 struct prefix_ipv4 p;
510 struct route_node *rn;
511 p.prefixlen = IPV4_MAX_PREFIXLEN;
512 p.prefix = addr;
513 rn = route_node_lookup (IF_OIFS_PARAMS (ifp), (struct prefix*)&p);
514 if (!rn || !rn->info)
515 return;
516
517 oip = rn->info;
518 route_unlock_node (rn);
519
520 if (!OSPF_IF_PARAM_CONFIGURED (oip, output_cost_cmd) &&
521 !OSPF_IF_PARAM_CONFIGURED (oip, transmit_delay) &&
522 !OSPF_IF_PARAM_CONFIGURED (oip, retransmit_interval) &&
523 !OSPF_IF_PARAM_CONFIGURED (oip, passive_interface) &&
524 !OSPF_IF_PARAM_CONFIGURED (oip, v_hello) &&
525 !OSPF_IF_PARAM_CONFIGURED (oip, v_wait) &&
526 !OSPF_IF_PARAM_CONFIGURED (oip, priority) &&
527 !OSPF_IF_PARAM_CONFIGURED (oip, type) &&
528 !OSPF_IF_PARAM_CONFIGURED (oip, auth_simple) &&
529 !OSPF_IF_PARAM_CONFIGURED (oip, auth_type) &&
530 listcount (oip->auth_crypt) == 0)
531 {
532 ospf_del_if_params (oip);
533 rn->info = NULL;
534 route_unlock_node (rn);
535 }
536}
537
538struct ospf_if_params *
539ospf_lookup_if_params (struct interface *ifp, struct in_addr addr)
540{
541 struct prefix_ipv4 p;
542 struct route_node *rn;
543
544 p.prefixlen = IPV4_MAX_PREFIXLEN;
545 p.prefix = addr;
546
547 rn = route_node_lookup (IF_OIFS_PARAMS (ifp), (struct prefix*)&p);
548
549 if (rn)
550 {
551 route_unlock_node (rn);
552 return rn->info;
553 }
554
555 return NULL;
556}
557
558struct ospf_if_params *
559ospf_get_if_params (struct interface *ifp, struct in_addr addr)
560{
561 struct prefix_ipv4 p;
562 struct route_node *rn;
563
564 p.family = AF_INET;
565 p.prefixlen = IPV4_MAX_PREFIXLEN;
566 p.prefix = addr;
567
568 rn = route_node_get (IF_OIFS_PARAMS (ifp), (struct prefix*)&p);
569
570 if (rn->info == NULL)
571 rn->info = ospf_new_if_params ();
572 else
573 route_unlock_node (rn);
574
575 return rn->info;
576}
577
578void
579ospf_if_update_params (struct interface *ifp, struct in_addr addr)
580{
581 struct route_node *rn;
582 struct ospf_interface *oi;
583
584 for (rn = route_top (IF_OIFS (ifp)); rn; rn = route_next (rn))
585 {
586 if ((oi = rn->info) == NULL)
587 continue;
588
589 if (IPV4_ADDR_SAME (&oi->address->u.prefix4, &addr))
590 oi->params = ospf_lookup_if_params (ifp, oi->address->u.prefix4);
591 }
592}
593
594int
595ospf_if_new_hook (struct interface *ifp)
596{
597 int rc = 0;
598
599 ifp->info = XMALLOC (MTYPE_OSPF_IF_INFO, sizeof (struct ospf_if_info));
600 memset (ifp->info, 0, sizeof (struct ospf_if_info));
601
602 IF_OIFS (ifp) = route_table_init ();
603 IF_OIFS_PARAMS (ifp) = route_table_init ();
604
605 IF_DEF_PARAMS (ifp) = ospf_new_if_params ();
606
607 SET_IF_PARAM (IF_DEF_PARAMS (ifp), transmit_delay);
608 IF_DEF_PARAMS (ifp)->transmit_delay = OSPF_TRANSMIT_DELAY_DEFAULT;
609
610 SET_IF_PARAM (IF_DEF_PARAMS (ifp), retransmit_interval);
611 IF_DEF_PARAMS (ifp)->retransmit_interval = OSPF_RETRANSMIT_INTERVAL_DEFAULT;
612
613 SET_IF_PARAM (IF_DEF_PARAMS (ifp), priority);
614 IF_DEF_PARAMS (ifp)->priority = OSPF_ROUTER_PRIORITY_DEFAULT;
615
616 SET_IF_PARAM (IF_DEF_PARAMS (ifp), passive_interface);
617 IF_DEF_PARAMS (ifp)->passive_interface = OSPF_IF_ACTIVE;
618
619 SET_IF_PARAM (IF_DEF_PARAMS (ifp), v_hello);
620 IF_DEF_PARAMS (ifp)->v_hello = OSPF_HELLO_INTERVAL_DEFAULT;
621
622 SET_IF_PARAM (IF_DEF_PARAMS (ifp), v_wait);
623 IF_DEF_PARAMS (ifp)->v_wait = OSPF_ROUTER_DEAD_INTERVAL_DEFAULT;
624
625 SET_IF_PARAM (IF_DEF_PARAMS (ifp), auth_simple);
626 memset (IF_DEF_PARAMS (ifp)->auth_simple, 0, OSPF_AUTH_SIMPLE_SIZE);
627
paul718e3742002-12-13 20:15:29 +0000628 SET_IF_PARAM (IF_DEF_PARAMS (ifp), auth_type);
629 IF_DEF_PARAMS (ifp)->auth_type = OSPF_AUTH_NOTSET;
630
631#ifdef HAVE_OPAQUE_LSA
632 rc = ospf_opaque_new_if (ifp);
633#endif /* HAVE_OPAQUE_LSA */
634 return rc;
635}
636
637int
638ospf_if_delete_hook (struct interface *ifp)
639{
640 int rc = 0;
641#ifdef HAVE_OPAQUE_LSA
642 rc = ospf_opaque_del_if (ifp);
643#endif /* HAVE_OPAQUE_LSA */
644 route_table_finish (IF_OIFS (ifp));
645 route_table_finish (IF_OIFS_PARAMS (ifp));
646 XFREE (MTYPE_OSPF_IF_INFO, ifp->info);
647 ifp->info = NULL;
648
649 return rc;
650}
651
652int
653ospf_if_is_enable (struct ospf_interface *oi)
654{
655 if (!if_is_loopback (oi->ifp))
656 if (if_is_up (oi->ifp))
657 return 1;
658
659 return 0;
660}
661
662int
663ospf_if_up (struct ospf_interface *oi)
664{
665 if (oi == NULL)
666 return 0;
667
668 if (oi->type == OSPF_IFTYPE_LOOPBACK)
669 OSPF_ISM_EVENT_SCHEDULE (oi, ISM_LoopInd);
670 else
671 {
672 if (oi->type != OSPF_IFTYPE_VIRTUALLINK)
paul68980082003-03-25 05:07:42 +0000673 ospf_if_add_allspfrouters (oi->ospf, oi->address, oi->ifp->ifindex);
paul718e3742002-12-13 20:15:29 +0000674 ospf_if_stream_set (oi);
675 OSPF_ISM_EVENT_SCHEDULE (oi, ISM_InterfaceUp);
676 }
677
678 return 1;
679}
680
681int
682ospf_if_down (struct ospf_interface *oi)
683{
684 if (oi == NULL)
685 return 0;
686
687 OSPF_ISM_EVENT_EXECUTE (oi, ISM_InterfaceDown);
688 /* Shutdown packet reception and sending */
689 ospf_if_stream_unset (oi);
690 if (oi->type != OSPF_IFTYPE_VIRTUALLINK)
paul68980082003-03-25 05:07:42 +0000691 ospf_if_drop_allspfrouters (oi->ospf, oi->address, oi->ifp->ifindex);
paul718e3742002-12-13 20:15:29 +0000692
693
694 return 1;
695}
696
697
698/* Virtual Link related functions. */
699
700struct ospf_vl_data *
701ospf_vl_data_new (struct ospf_area *area, struct in_addr vl_peer)
702{
703 struct ospf_vl_data *vl_data;
704
705 vl_data = XMALLOC (MTYPE_OSPF_VL_DATA, sizeof (struct ospf_vl_data));
706 memset (vl_data, 0, sizeof (struct ospf_vl_data));
707
708 vl_data->vl_peer.s_addr = vl_peer.s_addr;
709 vl_data->vl_area_id = area->area_id;
710 vl_data->format = area->format;
711
712 return vl_data;
713}
714
715void
716ospf_vl_data_free (struct ospf_vl_data *vl_data)
717{
718 XFREE (MTYPE_OSPF_VL_DATA, vl_data);
719}
720
721u_int vlink_count = 0;
722
723struct ospf_interface *
paul68980082003-03-25 05:07:42 +0000724ospf_vl_new (struct ospf *ospf, struct ospf_vl_data *vl_data)
paul718e3742002-12-13 20:15:29 +0000725{
726 struct ospf_interface * voi;
727 struct interface * vi;
728 char ifname[INTERFACE_NAMSIZ + 1];
729 struct ospf_area *area;
730 struct in_addr area_id;
731 struct connected *co;
732 struct prefix_ipv4 *p;
733
734 if (IS_DEBUG_OSPF_EVENT)
735 zlog_info ("ospf_vl_new(): Start");
736 if (vlink_count == OSPF_VL_MAX_COUNT)
737 {
738 if (IS_DEBUG_OSPF_EVENT)
739 zlog_info ("ospf_vl_new(): Alarm: "
740 "cannot create more than OSPF_MAX_VL_COUNT virtual links");
741 return NULL;
742 }
743
744 if (IS_DEBUG_OSPF_EVENT)
745 zlog_info ("ospf_vl_new(): creating pseudo zebra interface");
746
747 vi = if_create ();
748 co = connected_new ();
749 co->ifp = vi;
750 listnode_add (vi->connected, co);
751
752 p = prefix_ipv4_new ();
753 p->family = AF_INET;
754 p->prefix.s_addr = 0;
755 p->prefixlen = 0;
756
757 co->address = (struct prefix *)p;
758
paul68980082003-03-25 05:07:42 +0000759 voi = ospf_if_new (ospf, vi, co->address);
paul718e3742002-12-13 20:15:29 +0000760 if (voi == NULL)
761 {
762 if (IS_DEBUG_OSPF_EVENT)
763 zlog_info ("ospf_vl_new(): Alarm: OSPF int structure is not created");
764 return NULL;
765 }
766 voi->connected = co;
767 voi->vl_data = vl_data;
768 voi->ifp->mtu = OSPF_VL_MTU;
769 voi->type = OSPF_IFTYPE_VIRTUALLINK;
770
771 sprintf (ifname, "VLINK%d", vlink_count++);
772 if (IS_DEBUG_OSPF_EVENT)
773 zlog_info ("ospf_vl_new(): Created name: %s", ifname);
774 strncpy (vi->name, ifname, IFNAMSIZ);
775 if (IS_DEBUG_OSPF_EVENT)
776 zlog_info ("ospf_vl_new(): set if->name to %s", vi->name);
777
778 area_id.s_addr = 0;
paul68980082003-03-25 05:07:42 +0000779 area = ospf_area_get (ospf, area_id, OSPF_AREA_ID_FORMAT_ADDRESS);
paul718e3742002-12-13 20:15:29 +0000780 voi->area = area;
781
782 if (IS_DEBUG_OSPF_EVENT)
783 zlog_info ("ospf_vl_new(): set associated area to the backbone");
784
785 ospf_area_add_if (voi->area, voi);
786
787 ospf_if_stream_set (voi);
788
789 if (IS_DEBUG_OSPF_EVENT)
790 zlog_info ("ospf_vl_new(): Stop");
791 return voi;
792}
793
794void
795ospf_vl_if_delete (struct ospf_vl_data *vl_data)
796{
797 struct interface *ifp = vl_data->vl_oi->ifp;
798 vl_data->vl_oi->address->u.prefix4.s_addr = 0;
799 vl_data->vl_oi->address->prefixlen = 0;
800 ospf_if_free (vl_data->vl_oi);
801 if_delete (ifp);
802 vlink_count--;
803}
804
805struct ospf_vl_data *
806ospf_vl_lookup (struct ospf_area *area, struct in_addr vl_peer)
807{
808 struct ospf_vl_data *vl_data;
809 listnode node;
810
paul68980082003-03-25 05:07:42 +0000811 for (node = listhead (area->ospf->vlinks); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +0000812 if ((vl_data = getdata (node)) != NULL)
813 if (vl_data->vl_peer.s_addr == vl_peer.s_addr &&
814 IPV4_ADDR_SAME (&vl_data->vl_area_id, &area->area_id))
815 return vl_data;
816
817 return NULL;
818}
819
820void
821ospf_vl_shutdown (struct ospf_vl_data *vl_data)
822{
823 struct ospf_interface *oi;
824
825 if ((oi = vl_data->vl_oi) == NULL)
826 return;
827
828 oi->address->u.prefix4.s_addr = 0;
829 oi->address->prefixlen = 0;
830
831 UNSET_FLAG (oi->ifp->flags, IFF_UP);
832 /* OSPF_ISM_EVENT_SCHEDULE (oi, ISM_InterfaceDown); */
833 OSPF_ISM_EVENT_EXECUTE (oi, ISM_InterfaceDown);
834}
835
836void
paul68980082003-03-25 05:07:42 +0000837ospf_vl_add (struct ospf *ospf, struct ospf_vl_data *vl_data)
paul718e3742002-12-13 20:15:29 +0000838{
paul68980082003-03-25 05:07:42 +0000839 listnode_add (ospf->vlinks, vl_data);
paul718e3742002-12-13 20:15:29 +0000840#ifdef HAVE_SNMP
841 ospf_snmp_vl_add (vl_data);
842#endif /* HAVE_SNMP */
843}
844
845void
paul68980082003-03-25 05:07:42 +0000846ospf_vl_delete (struct ospf *ospf, struct ospf_vl_data *vl_data)
paul718e3742002-12-13 20:15:29 +0000847{
848 ospf_vl_shutdown (vl_data);
849 ospf_vl_if_delete (vl_data);
850
851#ifdef HAVE_SNMP
852 ospf_snmp_vl_delete (vl_data);
853#endif /* HAVE_SNMP */
paul68980082003-03-25 05:07:42 +0000854 listnode_delete (ospf->vlinks, vl_data);
paul718e3742002-12-13 20:15:29 +0000855
856 ospf_vl_data_free (vl_data);
857}
858
859void
860ospf_vl_set_params (struct ospf_vl_data *vl_data, struct vertex *v)
861{
862 int changed = 0;
863 struct ospf_interface *voi;
864 listnode node;
865 struct vertex_nexthop *nh;
866 int i;
867 struct router_lsa *rl;
868
869 voi = vl_data->vl_oi;
870
871 if (voi->output_cost != v->distance)
872 {
873 voi->output_cost = v->distance;
874 changed = 1;
875 }
876
877 for (node = listhead (v->nexthop); node; nextnode (node))
878 if ((nh = getdata (node)) != NULL)
879 {
880 vl_data->out_oi = (struct ospf_interface *) nh->oi;
881
882 voi->address->u.prefix4 = vl_data->out_oi->address->u.prefix4;
883 voi->address->prefixlen = vl_data->out_oi->address->prefixlen;
884
885 break; /* We take the first interface. */
886 }
887
888 rl = (struct router_lsa *)v->lsa;
889
890 for (i = 0; i < ntohs (rl->links); i++)
891 {
892 switch (rl->link[i].type)
893 {
894 case LSA_LINK_TYPE_VIRTUALLINK:
895 if (IS_DEBUG_OSPF_EVENT)
896 zlog_info ("found back link through VL");
897 case LSA_LINK_TYPE_TRANSIT:
898 case LSA_LINK_TYPE_POINTOPOINT:
899 vl_data->peer_addr = rl->link[i].link_data;
900 if (IS_DEBUG_OSPF_EVENT)
901 zlog_info ("%s peer address is %s\n",
902 vl_data->vl_oi->ifp->name, inet_ntoa(vl_data->peer_addr));
903 return;
904 }
905 }
906}
907
908
909void
paul68980082003-03-25 05:07:42 +0000910ospf_vl_up_check (struct ospf_area *area, struct in_addr rid,
paul718e3742002-12-13 20:15:29 +0000911 struct vertex *v)
912{
paul68980082003-03-25 05:07:42 +0000913 struct ospf *ospf = area->ospf;
paul718e3742002-12-13 20:15:29 +0000914 listnode node;
915 struct ospf_vl_data *vl_data;
916 struct ospf_interface *oi;
917
918 if (IS_DEBUG_OSPF_EVENT)
919 {
920 zlog_info ("ospf_vl_up_check(): Start");
921 zlog_info ("ospf_vl_up_check(): Router ID is %s", inet_ntoa (rid));
922 zlog_info ("ospf_vl_up_check(): Area is %s", inet_ntoa (area->area_id));
923 }
924
paul68980082003-03-25 05:07:42 +0000925 for (node = listhead (ospf->vlinks); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +0000926 {
927 if ((vl_data = getdata (node)) == NULL)
928 continue;
929
930 if (IS_DEBUG_OSPF_EVENT)
931 {
932 zlog_info ("ospf_vl_up_check(): considering VL, name: %s",
933 vl_data->vl_oi->ifp->name);
934 zlog_info ("ospf_vl_up_check(): VL area: %s, peer ID: %s",
935 inet_ntoa (vl_data->vl_area_id),
936 inet_ntoa (vl_data->vl_peer));
937 }
938
939 if (IPV4_ADDR_SAME (&vl_data->vl_peer, &rid) &&
940 IPV4_ADDR_SAME (&vl_data->vl_area_id, &area->area_id))
941 {
942 oi = vl_data->vl_oi;
943 SET_FLAG (vl_data->flags, OSPF_VL_FLAG_APPROVED);
944
945 if (IS_DEBUG_OSPF_EVENT)
946 zlog_info ("ospf_vl_up_check(): this VL matched");
947
948 if (oi->state == ISM_Down)
949 {
950 if (IS_DEBUG_OSPF_EVENT)
951 zlog_info ("ospf_vl_up_check(): VL is down, waking it up");
952 SET_FLAG (oi->ifp->flags, IFF_UP);
953 OSPF_ISM_EVENT_SCHEDULE (oi, ISM_InterfaceUp);
954 }
955
956 ospf_vl_set_params (vl_data, v);
957 }
958 }
959}
960
961void
paul68980082003-03-25 05:07:42 +0000962ospf_vl_unapprove (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +0000963{
964 listnode node;
965 struct ospf_vl_data *vl_data;
966
paul68980082003-03-25 05:07:42 +0000967 for (node = listhead (ospf->vlinks); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +0000968 if ((vl_data = getdata (node)) != NULL)
969 UNSET_FLAG (vl_data->flags, OSPF_VL_FLAG_APPROVED);
970}
971
972void
paul68980082003-03-25 05:07:42 +0000973ospf_vl_shut_unapproved (struct ospf *ospf)
paul718e3742002-12-13 20:15:29 +0000974{
975 listnode node;
976 struct ospf_vl_data *vl_data;
977
paul68980082003-03-25 05:07:42 +0000978 for (node = listhead (ospf->vlinks); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +0000979 if ((vl_data = getdata (node)) != NULL)
980 if (!CHECK_FLAG (vl_data->flags, OSPF_VL_FLAG_APPROVED))
981 ospf_vl_shutdown (vl_data);
982}
983
984int
985ospf_full_virtual_nbrs (struct ospf_area *area)
986{
987 if (IS_DEBUG_OSPF_EVENT)
988 {
989 zlog_info ("counting fully adjacent virtual neighbors in area %s",
990 inet_ntoa (area->area_id));
991 zlog_info ("there are %d of them", area->full_vls);
992 }
993
994 return area->full_vls;
995}
996
997int
998ospf_vls_in_area (struct ospf_area *area)
999{
1000 listnode node;
1001 struct ospf_vl_data *vl_data;
1002 int c = 0;
1003
paul68980082003-03-25 05:07:42 +00001004 for (node = listhead (area->ospf->vlinks); node; nextnode (node))
paul718e3742002-12-13 20:15:29 +00001005 if ((vl_data = getdata (node)) != NULL)
1006 if (IPV4_ADDR_SAME (&vl_data->vl_area_id, &area->area_id))
1007 c++;
1008
1009 return c;
1010}
1011
1012
1013struct crypt_key *
1014ospf_crypt_key_new ()
1015{
1016 struct crypt_key *ck;
1017
1018 ck = XMALLOC (MTYPE_OSPF_CRYPT_KEY, sizeof (struct crypt_key));
1019 memset (ck, 0, sizeof (struct crypt_key));
1020
1021 return ck;
1022}
1023
1024void
1025ospf_crypt_key_add (list crypt, struct crypt_key *ck)
1026{
1027 listnode_add (crypt, ck);
1028}
1029
1030struct crypt_key *
1031ospf_crypt_key_lookup (list auth_crypt, u_char key_id)
1032{
1033 listnode node;
1034 struct crypt_key *ck;
1035
1036 for (node = listhead (auth_crypt); node; nextnode (node))
1037 {
1038 ck = getdata (node);
1039 if (ck->key_id == key_id)
1040 return ck;
1041 }
1042
1043 return NULL;
1044}
1045
1046int
1047ospf_crypt_key_delete (list auth_crypt, u_char key_id)
1048{
1049 listnode node;
1050 struct crypt_key *ck;
1051
1052 for (node = listhead (auth_crypt); node; nextnode (node))
1053 {
1054 ck = getdata (node);
1055 if (ck->key_id == key_id)
1056 {
1057 listnode_delete (auth_crypt, ck);
1058 return 1;
1059 }
1060 }
1061
1062 return 0;
1063}
1064
1065void
1066ospf_if_init ()
1067{
1068 /* Initialize Zebra interface data structure. */
1069 if_init ();
paul020709f2003-04-04 02:44:16 +00001070 om->iflist = iflist;
paul718e3742002-12-13 20:15:29 +00001071 if_add_hook (IF_NEW_HOOK, ospf_if_new_hook);
1072 if_add_hook (IF_DELETE_HOOK, ospf_if_delete_hook);
1073}