]> git.proxmox.com Git - mirror_frr.git/blame - ospfd/ospfd.c
Merge pull request #1599 from chiragshah6/mdev
[mirror_frr.git] / ospfd / ospfd.c
CommitLineData
718e3744 1/* OSPF version 2 daemon program.
896014f4
DL
2 * Copyright (C) 1999, 2000 Toshiaki Takada
3 *
4 * This file is part of GNU Zebra.
5 *
6 * GNU Zebra is free software; you can redistribute it and/or modify it
7 * under the terms of the GNU General Public License as published by the
8 * Free Software Foundation; either version 2, or (at your option) any
9 * later version.
10 *
11 * GNU Zebra is distributed in the hope that it will be useful, but
12 * WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * General Public License for more details.
15 *
16 * You should have received a copy of the GNU General Public License along
17 * with this program; see the file COPYING; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19 */
718e3744 20
21#include <zebra.h>
22
23#include "thread.h"
24#include "vty.h"
25#include "command.h"
26#include "linklist.h"
27#include "prefix.h"
28#include "table.h"
29#include "if.h"
30#include "memory.h"
31#include "stream.h"
32#include "log.h"
d62a17ae 33#include "sockunion.h" /* for inet_aton () */
718e3744 34#include "zclient.h"
35#include "plist.h"
f102e75f 36#include "sockopt.h"
567b877d 37#include "bfd.h"
8879bd22 38#include "libfrr.h"
8efe88ea 39#include "defaults.h"
718e3744 40
41#include "ospfd/ospfd.h"
42#include "ospfd/ospf_network.h"
43#include "ospfd/ospf_interface.h"
44#include "ospfd/ospf_ism.h"
45#include "ospfd/ospf_asbr.h"
46#include "ospfd/ospf_lsa.h"
47#include "ospfd/ospf_lsdb.h"
48#include "ospfd/ospf_neighbor.h"
49#include "ospfd/ospf_nsm.h"
50#include "ospfd/ospf_spf.h"
51#include "ospfd/ospf_packet.h"
52#include "ospfd/ospf_dump.h"
53#include "ospfd/ospf_zebra.h"
54#include "ospfd/ospf_abr.h"
55#include "ospfd/ospf_flood.h"
56#include "ospfd/ospf_route.h"
57#include "ospfd/ospf_ase.h"
58
6b0655a2 59
ae19c240 60DEFINE_QOBJ_TYPE(ospf)
edd7c245 61
020709f9 62/* OSPF process wide configuration. */
63static struct ospf_master ospf_master;
64
65/* OSPF process wide configuration pointer to export. */
66struct ospf_master *om;
718e3744 67
68extern struct zclient *zclient;
69
6b0655a2 70
d62a17ae 71static void ospf_remove_vls_through_area(struct ospf *, struct ospf_area *);
72static void ospf_network_free(struct ospf *, struct ospf_network *);
73static void ospf_area_free(struct ospf_area *);
74static void ospf_network_run(struct prefix *, struct ospf_area *);
75static void ospf_network_run_interface(struct ospf *, struct interface *,
76 struct prefix *, struct ospf_area *);
77static void ospf_network_run_subnet(struct ospf *, struct connected *,
78 struct prefix *, struct ospf_area *);
79static int ospf_network_match_iface(const struct connected *,
80 const struct prefix *);
81static void ospf_finish_final(struct ospf *);
718e3744 82
718e3744 83#define OSPF_EXTERNAL_LSA_ORIGINATE_DELAY 1
6b0655a2 84
d62a17ae 85void ospf_router_id_update(struct ospf *ospf)
718e3744 86{
f4e14fdb 87 struct vrf *vrf = vrf_lookup_by_id(ospf->vrf_id);
d62a17ae 88 struct in_addr router_id, router_id_old;
89 struct ospf_interface *oi;
90 struct interface *ifp;
91 struct listnode *node;
92 int type;
2c19a6ec 93
d62a17ae 94 if (!ospf->oi_running) {
95 if (IS_DEBUG_OSPF_EVENT)
96 zlog_debug(
97 "Router ospf not configured -- Router-ID update postponed");
98 return;
25a346eb 99 }
718e3744 100
d62a17ae 101 if (IS_DEBUG_OSPF_EVENT)
102 zlog_debug("Router-ID[OLD:%s]: Update",
103 inet_ntoa(ospf->router_id));
104
105 router_id_old = ospf->router_id;
106
107 /* Select the router ID based on these priorities:
108 1. Statically assigned router ID is always the first choice.
109 2. If there is no statically assigned router ID, then try to stick
110 with the most recent value, since changing router ID's is very
111 disruptive.
112 3. Last choice: just go with whatever the zebra daemon recommends.
113 */
114 if (ospf->router_id_static.s_addr != 0)
115 router_id = ospf->router_id_static;
116 else if (ospf->router_id.s_addr != 0)
117 router_id = ospf->router_id;
118 else
6021c6c0 119 router_id = ospf->router_id_zebra;
d62a17ae 120
b5a8894d
CS
121 if (IS_DEBUG_OSPF_EVENT)
122 zlog_debug("Router-ID[OLD:%s]: Update to %s",
123 inet_ntoa(ospf->router_id),
6021c6c0 124 inet_ntoa(router_id));
d62a17ae 125
126 if (!IPV4_ADDR_SAME(&router_id_old, &router_id)) {
127
128 for (ALL_LIST_ELEMENTS_RO(ospf->oiflist, node, oi)) {
129 /* Some nbrs are identified by router_id, these needs
130 * to be rebuilt. Possible optimization would be to do
131 * oi->nbr_self->router_id = router_id for
132 * !(virtual | ptop) links
133 */
134 ospf_nbr_self_reset(oi, router_id);
135 }
136
137 /* If AS-external-LSA is queued, then flush those LSAs. */
138 if (router_id_old.s_addr == 0 && ospf->external_origin) {
139 /* Originate each redistributed external route. */
140 for (type = 0; type < ZEBRA_ROUTE_MAX; type++)
141 if (ospf->external_origin & (1 << type))
142 thread_add_event(
143 master,
144 ospf_external_lsa_originate_timer,
145 ospf, type, NULL);
146 /* Originate Deafult. */
147 if (ospf->external_origin & (1 << ZEBRA_ROUTE_MAX))
148 thread_add_event(master,
149 ospf_default_originate_timer,
150 ospf, 0, NULL);
151
152 ospf->external_origin = 0;
153 }
154
155 /* Flush (inline) all external LSAs based on the OSPF_LSA_SELF
156 * flag */
157 if (ospf->lsdb) {
158 struct route_node *rn;
159 struct ospf_lsa *lsa;
160
161 LSDB_LOOP(EXTERNAL_LSDB(ospf), rn, lsa)
162 if (IS_LSA_SELF(lsa))
163 ospf_lsa_flush_schedule(ospf, lsa);
164 }
165
166 ospf->router_id = router_id;
167 if (IS_DEBUG_OSPF_EVENT)
168 zlog_debug("Router-ID[NEW:%s]: Update",
169 inet_ntoa(ospf->router_id));
170
171 /* Flush (inline) all external LSAs which now match the new
172 router-id,
173 need to adjust the OSPF_LSA_SELF flag, so the flush doesnt
174 hit
175 asserts in ospf_refresher_unregister_lsa(). This step is
176 needed
177 because the current quagga code does look-up for
178 self-originated LSAs
179 based on the self router-id alone but expects OSPF_LSA_SELF
180 to be
181 properly set */
182 if (ospf->lsdb) {
183 struct route_node *rn;
184 struct ospf_lsa *lsa;
185
186 LSDB_LOOP(EXTERNAL_LSDB(ospf), rn, lsa)
187 {
188 /* AdvRouter and Router ID is the same. */
189 if (IPV4_ADDR_SAME(&lsa->data->adv_router,
190 &ospf->router_id)) {
191 SET_FLAG(lsa->flags,
192 OSPF_LSA_SELF_CHECKED);
193 SET_FLAG(lsa->flags, OSPF_LSA_SELF);
194 ospf_lsa_flush_schedule(ospf, lsa);
195 }
196 }
197 }
198
199 /* Originate each redistributed external route. */
200 for (type = 0; type < ZEBRA_ROUTE_MAX; type++)
201 thread_add_event(master,
202 ospf_external_lsa_originate_timer,
203 ospf, type, NULL);
204 thread_add_event(master, ospf_default_originate_timer, ospf, 0,
205 NULL);
206
207 /* update router-lsa's for each area */
208 ospf_router_lsa_update(ospf);
209
210 /* update ospf_interface's */
451fda4f 211 FOR_ALL_INTERFACES (vrf, ifp)
d62a17ae 212 ospf_if_update(ospf, ifp);
718e3744 213 }
718e3744 214}
6b0655a2 215
718e3744 216/* For OSPF area sort by area id. */
d62a17ae 217static int ospf_area_id_cmp(struct ospf_area *a1, struct ospf_area *a2)
718e3744 218{
d62a17ae 219 if (ntohl(a1->area_id.s_addr) > ntohl(a2->area_id.s_addr))
220 return 1;
221 if (ntohl(a1->area_id.s_addr) < ntohl(a2->area_id.s_addr))
222 return -1;
223 return 0;
718e3744 224}
225
226/* Allocate new ospf structure. */
b5a8894d 227static struct ospf *ospf_new(u_short instance, const char *name)
718e3744 228{
d62a17ae 229 int i;
b5a8894d 230 struct vrf *vrf = NULL;
718e3744 231
d62a17ae 232 struct ospf *new = XCALLOC(MTYPE_OSPF_TOP, sizeof(struct ospf));
718e3744 233
d62a17ae 234 new->instance = instance;
235 new->router_id.s_addr = htonl(0);
236 new->router_id_static.s_addr = htonl(0);
718e3744 237
b5a8894d
CS
238 if (name) {
239 new->vrf_id = VRF_UNKNOWN;
240 /* Freed in ospf_finish_final */
241 new->name = XSTRDUP(MTYPE_OSPF_TOP, name);
242 vrf = vrf_lookup_by_name(new->name);
243 if (IS_DEBUG_OSPF_EVENT)
244 zlog_debug("%s: Create new ospf instance with vrf_name %s vrf_id %d",
245 __PRETTY_FUNCTION__, name, new->vrf_id);
246 if (vrf)
247 ospf_vrf_link(new, vrf);
248 } else {
249 new->vrf_id = VRF_DEFAULT;
250 vrf = vrf_lookup_by_id(VRF_DEFAULT);
251 ospf_vrf_link(new, vrf);
252 }
253 ospf_zebra_vrf_register(new);
254
d62a17ae 255 new->abr_type = OSPF_ABR_DEFAULT;
256 new->oiflist = list_new();
257 new->vlinks = list_new();
258 new->areas = list_new();
259 new->areas->cmp = (int (*)(void *, void *))ospf_area_id_cmp;
260 new->networks = route_table_init();
261 new->nbr_nbma = route_table_init();
718e3744 262
d62a17ae 263 new->lsdb = ospf_lsdb_new();
718e3744 264
d62a17ae 265 new->default_originate = DEFAULT_ORIGINATE_NONE;
718e3744 266
d62a17ae 267 new->passive_interface_default = OSPF_IF_ACTIVE;
4ba4fc85 268
d62a17ae 269 new->new_external_route = route_table_init();
270 new->old_external_route = route_table_init();
271 new->external_lsas = route_table_init();
272
273 new->stub_router_startup_time = OSPF_STUB_ROUTER_UNCONFIGURED;
274 new->stub_router_shutdown_time = OSPF_STUB_ROUTER_UNCONFIGURED;
275 new->stub_router_admin_set = OSPF_STUB_ROUTER_ADMINISTRATIVE_UNSET;
276
277 /* Distribute parameter init. */
278 for (i = 0; i <= ZEBRA_ROUTE_MAX; i++) {
279 new->dtag[i] = 0;
280 }
281 new->default_metric = -1;
282 new->ref_bandwidth = OSPF_DEFAULT_REF_BANDWIDTH;
283
284 /* LSA timers */
285 new->min_ls_interval = OSPF_MIN_LS_INTERVAL;
286 new->min_ls_arrival = OSPF_MIN_LS_ARRIVAL;
287
288 /* SPF timer value init. */
289 new->spf_delay = OSPF_SPF_DELAY_DEFAULT;
290 new->spf_holdtime = OSPF_SPF_HOLDTIME_DEFAULT;
291 new->spf_max_holdtime = OSPF_SPF_MAX_HOLDTIME_DEFAULT;
292 new->spf_hold_multiplier = 1;
293
294 /* MaxAge init. */
295 new->maxage_delay = OSPF_LSA_MAXAGE_REMOVE_DELAY_DEFAULT;
296 new->maxage_lsa = route_table_init();
297 new->t_maxage_walker = NULL;
298 thread_add_timer(master, ospf_lsa_maxage_walker, new,
299 OSPF_LSA_MAXAGE_CHECK_INTERVAL, &new->t_maxage_walker);
300
301 /* Distance table init. */
302 new->distance_table = route_table_init();
303
304 new->lsa_refresh_queue.index = 0;
305 new->lsa_refresh_interval = OSPF_LSA_REFRESH_INTERVAL_DEFAULT;
306 new->t_lsa_refresher = NULL;
307 thread_add_timer(master, ospf_lsa_refresh_walker, new,
308 new->lsa_refresh_interval, &new->t_lsa_refresher);
309 new->lsa_refresher_started = monotime(NULL);
310
e7503eab 311 if ((ospf_sock_init(new)) < 0) {
d62a17ae 312 zlog_err(
313 "ospf_new: fatal error: ospf_sock_init was unable to open "
314 "a socket");
315 exit(1);
316 }
317 if ((new->ibuf = stream_new(OSPF_MAX_PACKET_SIZE + 1)) == NULL) {
318 zlog_err(
319 "ospf_new: fatal error: stream_new(%u) failed allocating ibuf",
320 OSPF_MAX_PACKET_SIZE + 1);
321 exit(1);
322 }
323 new->t_read = NULL;
324 thread_add_read(master, ospf_read, new, new->fd, &new->t_read);
325 new->oi_write_q = list_new();
326 new->write_oi_count = OSPF_WRITE_INTERFACE_COUNT_DEFAULT;
327
328/* Enable "log-adjacency-changes" */
8efe88ea 329#if DFLT_OSPF_LOG_ADJACENCY_CHANGES
d62a17ae 330 SET_FLAG(new->config, OSPF_LOG_ADJACENCY_CHANGES);
8efe88ea 331#endif
3e7c8d04 332
d62a17ae 333 QOBJ_REG(new, ospf);
ae19c240 334
d62a17ae 335 return new;
718e3744 336}
337
d62a17ae 338struct ospf *ospf_lookup_instance(u_short instance)
7c8ff89e 339{
d62a17ae 340 struct ospf *ospf;
341 struct listnode *node, *nnode;
7c8ff89e 342
d62a17ae 343 if (listcount(om->ospf) == 0)
344 return NULL;
7c8ff89e 345
d62a17ae 346 for (ALL_LIST_ELEMENTS(om->ospf, node, nnode, ospf))
347 if ((ospf->instance == 0 && instance == 0)
348 || (ospf->instance && instance
349 && ospf->instance == instance))
350 return ospf;
7c8ff89e 351
d62a17ae 352 return NULL;
7c8ff89e
DS
353}
354
d62a17ae 355static int ospf_is_ready(struct ospf *ospf)
52c62ab8 356{
d62a17ae 357 /* OSPF must be on and Router-ID must be configured. */
358 if (!ospf || ospf->router_id.s_addr == 0)
359 return 0;
360
361 return 1;
52c62ab8
JAG
362}
363
d62a17ae 364static void ospf_add(struct ospf *ospf)
020709f9 365{
d62a17ae 366 listnode_add(om->ospf, ospf);
020709f9 367}
718e3744 368
d62a17ae 369static void ospf_delete(struct ospf *ospf)
020709f9 370{
d62a17ae 371 listnode_delete(om->ospf, ospf);
020709f9 372}
68980084 373
b5a8894d
CS
374struct ospf *ospf_lookup_by_inst_name(u_short instance, const char *name)
375{
376 struct ospf *ospf = NULL;
377 struct listnode *node, *nnode;
378
379 for (ALL_LIST_ELEMENTS(om->ospf, node, nnode, ospf)) {
380 if ((ospf->instance == instance) &&
381 ((ospf->name == NULL && name == NULL) ||
382 (ospf->name && name && strcmp(ospf->name, name) == 0)))
383 return ospf;
384 }
385 return NULL;
386}
387
388struct ospf *ospf_get(u_short instance, const char *name)
020709f9 389{
d62a17ae 390 struct ospf *ospf;
020709f9 391
b5a8894d
CS
392 /* vrf name provided call inst and name based api
393 * in case of no name pass default ospf instance */
394 if (name)
395 ospf = ospf_lookup_by_inst_name(instance, name);
396 else
397 ospf = ospf_lookup_by_vrf_id(VRF_DEFAULT);
398
d62a17ae 399 if (ospf == NULL) {
b5a8894d 400 ospf = ospf_new(instance, name);
d62a17ae 401 ospf_add(ospf);
7c8ff89e 402
d62a17ae 403 if (ospf->router_id_static.s_addr == 0)
404 ospf_router_id_update(ospf);
7c8ff89e 405
d62a17ae 406 ospf_opaque_type11_lsa_init(ospf);
407 }
7c8ff89e 408
d62a17ae 409 return ospf;
7c8ff89e
DS
410}
411
d62a17ae 412struct ospf *ospf_get_instance(u_short instance)
7c8ff89e 413{
d62a17ae 414 struct ospf *ospf;
7c8ff89e 415
d62a17ae 416 ospf = ospf_lookup_instance(instance);
417 if (ospf == NULL) {
b5a8894d 418 ospf = ospf_new(instance, NULL /* VRF_DEFAULT*/);
d62a17ae 419 ospf_add(ospf);
020709f9 420
b5a8894d
CS
421 if (ospf->router_id_static.s_addr == 0) {
422 if (vrf_lookup_by_id(ospf->vrf_id))
423 ospf_router_id_update(ospf);
424 else {
425 if (IS_DEBUG_OSPF_EVENT)
426 zlog_debug("%s: ospf VRF (id %d) is not active yet, skip router id update"
427 , __PRETTY_FUNCTION__,
428 ospf->vrf_id);
429 }
d62a17ae 430 ospf_router_id_update(ospf);
b5a8894d 431 }
718e3744 432
d62a17ae 433 ospf_opaque_type11_lsa_init(ospf);
434 }
68980084 435
d62a17ae 436 return ospf;
718e3744 437}
6b0655a2 438
b5a8894d
CS
439struct ospf *ospf_lookup_by_vrf_id(vrf_id_t vrf_id)
440{
441 struct vrf *vrf = NULL;
442
443 vrf = vrf_lookup_by_id(vrf_id);
444 if (!vrf)
445 return NULL;
446 return (vrf->info) ? (struct ospf *)vrf->info : NULL;
447
448}
449
43b8d1d8
CS
450/* It should only be used when processing incoming info update from zebra.
451 * Other situations, it is not sufficient to lookup the ospf instance by
452 * vrf_name only without using the instance number.
453 */
454static struct ospf *ospf_lookup_by_name(const char *vrf_name)
b5a8894d
CS
455{
456 struct ospf *ospf = NULL;
457 struct listnode *node, *nnode;
458
459 for (ALL_LIST_ELEMENTS(om->ospf, node, nnode, ospf))
43b8d1d8
CS
460 if ((ospf->name == NULL && vrf_name == NULL)
461 || (ospf->name && vrf_name &&
462 strcmp(ospf->name, vrf_name) == 0))
b5a8894d
CS
463 return ospf;
464 return NULL;
465}
466
c9c93d50 467/* Handle the second half of deferred shutdown. This is called either
468 * from the deferred-shutdown timer thread, or directly through
469 * ospf_deferred_shutdown_check.
88d6cf37 470 *
471 * Function is to cleanup G-R state, if required then call ospf_finish_final
472 * to complete shutdown of this ospf instance. Possibly exit if the
473 * whole process is being shutdown and this was the last OSPF instance.
474 */
d62a17ae 475static void ospf_deferred_shutdown_finish(struct ospf *ospf)
476{
477 ospf->stub_router_shutdown_time = OSPF_STUB_ROUTER_UNCONFIGURED;
478 OSPF_TIMER_OFF(ospf->t_deferred_shutdown);
479
480 ospf_finish_final(ospf);
481
482 /* *ospf is now invalid */
88d6cf37 483
d62a17ae 484 /* ospfd being shut-down? If so, was this the last ospf instance? */
485 if (CHECK_FLAG(om->options, OSPF_MASTER_SHUTDOWN)
486 && (listcount(om->ospf) == 0)) {
487 exit(0);
488 }
489
490 return;
88d6cf37 491}
492
493/* Timer thread for G-R */
d62a17ae 494static int ospf_deferred_shutdown_timer(struct thread *t)
88d6cf37 495{
d62a17ae 496 struct ospf *ospf = THREAD_ARG(t);
497
498 ospf_deferred_shutdown_finish(ospf);
499
500 return 0;
88d6cf37 501}
502
c9c93d50 503/* Check whether deferred-shutdown must be scheduled, otherwise call
88d6cf37 504 * down directly into second-half of instance shutdown.
505 */
d62a17ae 506static void ospf_deferred_shutdown_check(struct ospf *ospf)
507{
508 unsigned long timeout;
509 struct listnode *ln;
510 struct ospf_area *area;
511
512 /* deferred shutdown already running? */
513 if (ospf->t_deferred_shutdown)
514 return;
515
516 /* Should we try push out max-metric LSAs? */
517 if (ospf->stub_router_shutdown_time != OSPF_STUB_ROUTER_UNCONFIGURED) {
518 for (ALL_LIST_ELEMENTS_RO(ospf->areas, ln, area)) {
519 SET_FLAG(area->stub_router_state,
520 OSPF_AREA_ADMIN_STUB_ROUTED);
521
522 if (!CHECK_FLAG(area->stub_router_state,
523 OSPF_AREA_IS_STUB_ROUTED))
524 ospf_router_lsa_update_area(area);
525 }
526 timeout = ospf->stub_router_shutdown_time;
527 } else {
528 /* No timer needed */
529 ospf_deferred_shutdown_finish(ospf);
530 return;
531 }
532
533 OSPF_TIMER_ON(ospf->t_deferred_shutdown, ospf_deferred_shutdown_timer,
534 timeout);
535 return;
88d6cf37 536}
6b0655a2 537
88d6cf37 538/* Shut down the entire process */
d62a17ae 539void ospf_terminate(void)
540{
541 struct ospf *ospf;
542 struct listnode *node, *nnode;
543
544 /* shutdown already in progress */
545 if (CHECK_FLAG(om->options, OSPF_MASTER_SHUTDOWN))
546 return;
547
548 SET_FLAG(om->options, OSPF_MASTER_SHUTDOWN);
549
550 /* exit immediately if OSPF not actually running */
551 if (listcount(om->ospf) == 0)
552 exit(0);
553
554 bfd_gbl_exit();
555 for (ALL_LIST_ELEMENTS(om->ospf, node, nnode, ospf))
556 ospf_finish(ospf);
557
558 /* Deliberately go back up, hopefully to thread scheduler, as
559 * One or more ospf_finish()'s may have deferred shutdown to a timer
560 * thread
561 */
562 zclient_stop(zclient);
563 zclient_free(zclient);
8879bd22
RW
564
565 frr_fini();
d62a17ae 566}
567
568void ospf_finish(struct ospf *ospf)
569{
570 /* let deferred shutdown decide */
571 ospf_deferred_shutdown_check(ospf);
572
573 /* if ospf_deferred_shutdown returns, then ospf_finish_final is
574 * deferred to expiry of G-S timer thread. Return back up, hopefully
575 * to thread scheduler.
576 */
577 return;
88d6cf37 578}
579
580/* Final cleanup of ospf instance */
d62a17ae 581static void ospf_finish_final(struct ospf *ospf)
718e3744 582{
f4e14fdb 583 struct vrf *vrf = vrf_lookup_by_id(ospf->vrf_id);
d62a17ae 584 struct route_node *rn;
585 struct ospf_nbr_nbma *nbr_nbma;
586 struct ospf_lsa *lsa;
587 struct interface *ifp;
588 struct ospf_interface *oi;
589 struct ospf_area *area;
590 struct ospf_vl_data *vl_data;
591 struct listnode *node, *nnode;
592 int i;
593 u_short instance = 0;
7c8ff89e 594
d62a17ae 595 QOBJ_UNREG(ospf);
7c8ff89e 596
d62a17ae 597 ospf_opaque_type11_lsa_term(ospf);
953cde65 598
046460a1 599 ospf_flush_self_originated_lsas_now(ospf);
953cde65 600
d62a17ae 601 /* Unregister redistribution */
602 for (i = 0; i < ZEBRA_ROUTE_MAX; i++) {
603 struct list *red_list;
604 struct ospf_redist *red;
718e3744 605
d62a17ae 606 red_list = ospf->redist[i];
607 if (!red_list)
608 continue;
718e3744 609
d62a17ae 610 for (ALL_LIST_ELEMENTS(red_list, node, nnode, red))
611 ospf_redistribute_unset(ospf, i, red->instance);
612 }
613 ospf_redistribute_default_unset(ospf);
718e3744 614
d62a17ae 615 for (ALL_LIST_ELEMENTS(ospf->areas, node, nnode, area))
616 ospf_remove_vls_through_area(ospf, area);
718e3744 617
d62a17ae 618 for (ALL_LIST_ELEMENTS(ospf->vlinks, node, nnode, vl_data))
619 ospf_vl_delete(ospf, vl_data);
718e3744 620
affe9e99 621 list_delete_and_null(&ospf->vlinks);
718e3744 622
d62a17ae 623 /* Remove any ospf interface config params */
451fda4f 624 FOR_ALL_INTERFACES (vrf, ifp) {
d62a17ae 625 struct ospf_if_params *params;
718e3744 626
d62a17ae 627 params = IF_DEF_PARAMS(ifp);
628 if (OSPF_IF_PARAM_CONFIGURED(params, if_area))
629 UNSET_IF_PARAM(params, if_area);
718e3744 630 }
718e3744 631
d62a17ae 632 /* Reset interface. */
633 for (ALL_LIST_ELEMENTS(ospf->oiflist, node, nnode, oi))
634 ospf_if_free(oi);
84519c81 635 list_delete_and_null(&ospf->oiflist);
718e3744 636
b5a8894d
CS
637 /* De-Register VRF */
638 ospf_zebra_vrf_deregister(ospf);
639
d62a17ae 640 /* Clear static neighbors */
641 for (rn = route_top(ospf->nbr_nbma); rn; rn = route_next(rn))
642 if ((nbr_nbma = rn->info)) {
643 OSPF_POLL_TIMER_OFF(nbr_nbma->t_poll);
718e3744 644
d62a17ae 645 if (nbr_nbma->nbr) {
646 nbr_nbma->nbr->nbr_nbma = NULL;
647 nbr_nbma->nbr = NULL;
648 }
649
650 if (nbr_nbma->oi) {
651 listnode_delete(nbr_nbma->oi->nbr_nbma,
652 nbr_nbma);
653 nbr_nbma->oi = NULL;
654 }
655
656 XFREE(MTYPE_OSPF_NEIGHBOR_STATIC, nbr_nbma);
657 }
658
659 route_table_finish(ospf->nbr_nbma);
660
661 /* Clear networks and Areas. */
662 for (rn = route_top(ospf->networks); rn; rn = route_next(rn)) {
663 struct ospf_network *network;
664
665 if ((network = rn->info) != NULL) {
666 ospf_network_free(ospf, network);
667 rn->info = NULL;
668 route_unlock_node(rn);
669 }
91e6a0e5 670 }
7f586094 671 route_table_finish(ospf->networks);
718e3744 672
d62a17ae 673 for (ALL_LIST_ELEMENTS(ospf->areas, node, nnode, area)) {
674 listnode_delete(ospf->areas, area);
675 ospf_area_free(area);
676 }
718e3744 677
d62a17ae 678 /* Cancel all timers. */
679 OSPF_TIMER_OFF(ospf->t_external_lsa);
680 OSPF_TIMER_OFF(ospf->t_spf_calc);
681 OSPF_TIMER_OFF(ospf->t_ase_calc);
682 OSPF_TIMER_OFF(ospf->t_maxage);
683 OSPF_TIMER_OFF(ospf->t_maxage_walker);
684 OSPF_TIMER_OFF(ospf->t_abr_task);
685 OSPF_TIMER_OFF(ospf->t_asbr_check);
686 OSPF_TIMER_OFF(ospf->t_distribute_update);
687 OSPF_TIMER_OFF(ospf->t_lsa_refresher);
688 OSPF_TIMER_OFF(ospf->t_read);
689 OSPF_TIMER_OFF(ospf->t_write);
690 OSPF_TIMER_OFF(ospf->t_opaque_lsa_self);
691
692 close(ospf->fd);
693 stream_free(ospf->ibuf);
694
695 LSDB_LOOP(OPAQUE_AS_LSDB(ospf), rn, lsa)
696 ospf_discard_from_db(ospf, ospf->lsdb, lsa);
697 LSDB_LOOP(EXTERNAL_LSDB(ospf), rn, lsa)
698 ospf_discard_from_db(ospf, ospf->lsdb, lsa);
699
700 ospf_lsdb_delete_all(ospf->lsdb);
701 ospf_lsdb_free(ospf->lsdb);
702
703 for (rn = route_top(ospf->maxage_lsa); rn; rn = route_next(rn)) {
704 struct ospf_lsa *lsa;
705
706 if ((lsa = rn->info) != NULL) {
707 ospf_lsa_unlock(&lsa);
708 rn->info = NULL;
709 }
710 route_unlock_node(rn);
711 }
712 route_table_finish(ospf->maxage_lsa);
713
714 if (ospf->old_table)
715 ospf_route_table_free(ospf->old_table);
716 if (ospf->new_table) {
b5a8894d 717 ospf_route_delete(ospf, ospf->new_table);
d62a17ae 718 ospf_route_table_free(ospf->new_table);
719 }
720 if (ospf->old_rtrs)
721 ospf_rtrs_free(ospf->old_rtrs);
722 if (ospf->new_rtrs)
723 ospf_rtrs_free(ospf->new_rtrs);
724 if (ospf->new_external_route) {
b5a8894d 725 ospf_route_delete(ospf, ospf->new_external_route);
d62a17ae 726 ospf_route_table_free(ospf->new_external_route);
727 }
728 if (ospf->old_external_route) {
b5a8894d 729 ospf_route_delete(ospf, ospf->old_external_route);
d62a17ae 730 ospf_route_table_free(ospf->old_external_route);
731 }
732 if (ospf->external_lsas) {
733 ospf_ase_external_lsas_finish(ospf->external_lsas);
734 }
718e3744 735
affe9e99
DS
736 list_delete_and_null(&ospf->areas);
737 list_delete_and_null(&ospf->oi_write_q);
d62a17ae 738
739 for (i = ZEBRA_ROUTE_SYSTEM; i <= ZEBRA_ROUTE_MAX; i++) {
740 struct list *ext_list;
741 struct listnode *node;
742 struct ospf_external *ext;
743
de1ac5fd 744 ext_list = ospf->external[i];
d62a17ae 745 if (!ext_list)
746 continue;
747
748 for (ALL_LIST_ELEMENTS_RO(ext_list, node, ext)) {
749 if (ext->external_info)
750 for (rn = route_top(ext->external_info); rn;
751 rn = route_next(rn)) {
752 if (rn->info == NULL)
753 continue;
754
755 XFREE(MTYPE_OSPF_EXTERNAL_INFO,
756 rn->info);
757 rn->info = NULL;
758 route_unlock_node(rn);
759 }
760 }
761 }
718e3744 762
d62a17ae 763 ospf_distance_reset(ospf);
764 route_table_finish(ospf->distance_table);
7c8ff89e 765
d62a17ae 766 if (!CHECK_FLAG(om->options, OSPF_MASTER_SHUTDOWN))
767 instance = ospf->instance;
718e3744 768
d62a17ae 769 ospf_delete(ospf);
7c8ff89e 770
b5a8894d
CS
771 if (ospf->name) {
772 vrf = vrf_lookup_by_name(ospf->name);
773 if (vrf)
774 ospf_vrf_unlink(ospf, vrf);
775 XFREE(MTYPE_OSPF_TOP, ospf->name);
776 } else {
777 vrf = vrf_lookup_by_id(VRF_DEFAULT);
778 if (vrf)
779 ospf_vrf_unlink(ospf, vrf);
780 }
781
d62a17ae 782 XFREE(MTYPE_OSPF_TOP, ospf);
7c8ff89e 783
d62a17ae 784 if (!CHECK_FLAG(om->options, OSPF_MASTER_SHUTDOWN))
785 ospf_get_instance(instance);
718e3744 786}
787
6b0655a2 788
718e3744 789/* allocate new OSPF Area object */
d62a17ae 790static struct ospf_area *ospf_area_new(struct ospf *ospf,
791 struct in_addr area_id)
718e3744 792{
d62a17ae 793 struct ospf_area *new;
718e3744 794
d62a17ae 795 /* Allocate new config_network. */
796 new = XCALLOC(MTYPE_OSPF_AREA, sizeof(struct ospf_area));
718e3744 797
d62a17ae 798 new->ospf = ospf;
718e3744 799
d62a17ae 800 new->area_id = area_id;
801 new->area_id_fmt = OSPF_AREA_ID_FMT_DOTTEDQUAD;
718e3744 802
d62a17ae 803 new->external_routing = OSPF_AREA_DEFAULT;
804 new->default_cost = 1;
805 new->auth_type = OSPF_AUTH_NULL;
718e3744 806
d62a17ae 807 /* New LSDB init. */
808 new->lsdb = ospf_lsdb_new();
718e3744 809
d62a17ae 810 /* Self-originated LSAs initialize. */
811 new->router_lsa_self = NULL;
718e3744 812
d62a17ae 813 ospf_opaque_type10_lsa_init(new);
718e3744 814
d62a17ae 815 new->oiflist = list_new();
816 new->ranges = route_table_init();
718e3744 817
d62a17ae 818 if (area_id.s_addr == OSPF_AREA_BACKBONE)
819 ospf->backbone = new;
820
821 return new;
718e3744 822}
823
d62a17ae 824static void ospf_area_free(struct ospf_area *area)
718e3744 825{
d62a17ae 826 struct route_node *rn;
827 struct ospf_lsa *lsa;
828
7f7f77a0
DS
829 ospf_opaque_type10_lsa_term(area);
830
d62a17ae 831 /* Free LSDBs. */
832 LSDB_LOOP(ROUTER_LSDB(area), rn, lsa)
833 ospf_discard_from_db(area->ospf, area->lsdb, lsa);
834 LSDB_LOOP(NETWORK_LSDB(area), rn, lsa)
835 ospf_discard_from_db(area->ospf, area->lsdb, lsa);
836 LSDB_LOOP(SUMMARY_LSDB(area), rn, lsa)
837 ospf_discard_from_db(area->ospf, area->lsdb, lsa);
838 LSDB_LOOP(ASBR_SUMMARY_LSDB(area), rn, lsa)
839 ospf_discard_from_db(area->ospf, area->lsdb, lsa);
68980084 840
d62a17ae 841 LSDB_LOOP(NSSA_LSDB(area), rn, lsa)
842 ospf_discard_from_db(area->ospf, area->lsdb, lsa);
843 LSDB_LOOP(OPAQUE_AREA_LSDB(area), rn, lsa)
844 ospf_discard_from_db(area->ospf, area->lsdb, lsa);
845 LSDB_LOOP(OPAQUE_LINK_LSDB(area), rn, lsa)
846 ospf_discard_from_db(area->ospf, area->lsdb, lsa);
718e3744 847
7f586094 848 ospf_opaque_type10_lsa_term(area);
d62a17ae 849 ospf_lsdb_delete_all(area->lsdb);
850 ospf_lsdb_free(area->lsdb);
718e3744 851
d62a17ae 852 ospf_lsa_unlock(&area->router_lsa_self);
718e3744 853
d62a17ae 854 route_table_finish(area->ranges);
affe9e99 855 list_delete_and_null(&area->oiflist);
718e3744 856
d62a17ae 857 if (EXPORT_NAME(area))
858 free(EXPORT_NAME(area));
718e3744 859
d62a17ae 860 if (IMPORT_NAME(area))
861 free(IMPORT_NAME(area));
718e3744 862
d62a17ae 863 /* Cancel timer. */
864 OSPF_TIMER_OFF(area->t_stub_router);
865 OSPF_TIMER_OFF(area->t_opaque_lsa_self);
718e3744 866
d62a17ae 867 if (OSPF_IS_AREA_BACKBONE(area))
868 area->ospf->backbone = NULL;
869
870 XFREE(MTYPE_OSPF_AREA, area);
718e3744 871}
872
d62a17ae 873void ospf_area_check_free(struct ospf *ospf, struct in_addr area_id)
718e3744 874{
d62a17ae 875 struct ospf_area *area;
718e3744 876
d62a17ae 877 area = ospf_area_lookup_by_area_id(ospf, area_id);
878 if (area && listcount(area->oiflist) == 0 && area->ranges->top == NULL
879 && area->shortcut_configured == OSPF_SHORTCUT_DEFAULT
880 && area->external_routing == OSPF_AREA_DEFAULT
881 && area->no_summary == 0 && area->default_cost == 1
882 && EXPORT_NAME(area) == NULL && IMPORT_NAME(area) == NULL
883 && area->auth_type == OSPF_AUTH_NULL) {
884 listnode_delete(ospf->areas, area);
885 ospf_area_free(area);
886 }
718e3744 887}
888
d62a17ae 889struct ospf_area *ospf_area_get(struct ospf *ospf, struct in_addr area_id)
718e3744 890{
d62a17ae 891 struct ospf_area *area;
892
893 area = ospf_area_lookup_by_area_id(ospf, area_id);
894 if (!area) {
895 area = ospf_area_new(ospf, area_id);
896 listnode_add_sort(ospf->areas, area);
897 ospf_check_abr_status(ospf);
898 if (ospf->stub_router_admin_set
899 == OSPF_STUB_ROUTER_ADMINISTRATIVE_SET) {
900 SET_FLAG(area->stub_router_state,
901 OSPF_AREA_ADMIN_STUB_ROUTED);
902 }
903 }
718e3744 904
d62a17ae 905 return area;
718e3744 906}
907
d62a17ae 908struct ospf_area *ospf_area_lookup_by_area_id(struct ospf *ospf,
909 struct in_addr area_id)
718e3744 910{
d62a17ae 911 struct ospf_area *area;
912 struct listnode *node;
718e3744 913
d62a17ae 914 for (ALL_LIST_ELEMENTS_RO(ospf->areas, node, area))
915 if (IPV4_ADDR_SAME(&area->area_id, &area_id))
916 return area;
718e3744 917
d62a17ae 918 return NULL;
718e3744 919}
920
d62a17ae 921void ospf_area_add_if(struct ospf_area *area, struct ospf_interface *oi)
718e3744 922{
d62a17ae 923 listnode_add(area->oiflist, oi);
718e3744 924}
925
d62a17ae 926void ospf_area_del_if(struct ospf_area *area, struct ospf_interface *oi)
718e3744 927{
d62a17ae 928 listnode_delete(area->oiflist, oi);
718e3744 929}
930
52c62ab8 931
d62a17ae 932static void add_ospf_interface(struct connected *co, struct ospf_area *area)
953cde65 933{
d62a17ae 934 struct ospf_interface *oi;
953cde65 935
d62a17ae 936 oi = ospf_if_new(area->ospf, co->ifp, co->address);
937 oi->connected = co;
953cde65 938
d62a17ae 939 oi->area = area;
953cde65 940
d62a17ae 941 oi->params = ospf_lookup_if_params(co->ifp, oi->address->u.prefix4);
942 oi->output_cost = ospf_if_get_output_cost(oi);
953cde65 943
d62a17ae 944 /* Relate ospf interface to ospf instance. */
945 oi->ospf = area->ospf;
953cde65 946
d62a17ae 947 /* update network type as interface flag */
948 /* If network type is specified previously,
949 skip network type setting. */
950 oi->type = IF_DEF_PARAMS(co->ifp)->type;
953cde65 951
d62a17ae 952 /* Add pseudo neighbor. */
953 ospf_nbr_self_reset(oi, oi->ospf->router_id);
953cde65 954
d62a17ae 955 ospf_area_add_if(oi->area, oi);
52c62ab8 956
d62a17ae 957 /*
958 * if router_id is not configured, dont bring up
959 * interfaces.
960 * ospf_router_id_update() will call ospf_if_update
961 * whenever r-id is configured instead.
962 */
963 if ((area->ospf->router_id.s_addr != 0) && if_is_operative(co->ifp))
964 ospf_if_up(oi);
953cde65
JT
965}
966
967static void update_redistributed(struct ospf *ospf, int add_to_ospf)
968{
d62a17ae 969 struct route_node *rn;
970 struct external_info *ei;
971 struct ospf_external *ext;
972
de1ac5fd
CS
973 if (ospf_is_type_redistributed(ospf, ZEBRA_ROUTE_CONNECT, 0)) {
974 ext = ospf_external_lookup(ospf, ZEBRA_ROUTE_CONNECT, 0);
975 if ((ext) && EXTERNAL_INFO(ext)) {
d62a17ae 976 for (rn = route_top(EXTERNAL_INFO(ext)); rn;
977 rn = route_next(rn)) {
de1ac5fd
CS
978 ei = rn->info;
979 if (ei == NULL)
980 continue;
981
982 if (add_to_ospf) {
983 if (ospf_external_info_find_lsa(
984 ospf, &ei->p))
985 if (!ospf_distribute_check_connected(
986 ospf, ei))
987 ospf_external_lsa_flush(
988 ospf,
989 ei->type,
990 &ei->p,
991 ei->ifindex /*, ei->nexthop */);
992 } else {
993 if (!ospf_external_info_find_lsa(
994 ospf, &ei->p))
995 if (ospf_distribute_check_connected(
996 ospf, ei))
997 ospf_external_lsa_originate(
998 ospf,
999 ei);
d62a17ae 1000 }
1001 }
1002 }
de1ac5fd 1003 }
953cde65
JT
1004}
1005
718e3744 1006/* Config network statement related functions. */
d62a17ae 1007static struct ospf_network *ospf_network_new(struct in_addr area_id)
718e3744 1008{
d62a17ae 1009 struct ospf_network *new;
1010 new = XCALLOC(MTYPE_OSPF_NETWORK, sizeof(struct ospf_network));
718e3744 1011
d62a17ae 1012 new->area_id = area_id;
1013 new->area_id_fmt = OSPF_AREA_ID_FMT_DOTTEDQUAD;
1014
1015 return new;
718e3744 1016}
1017
d62a17ae 1018static void ospf_network_free(struct ospf *ospf, struct ospf_network *network)
718e3744 1019{
d62a17ae 1020 ospf_area_check_free(ospf, network->area_id);
1021 ospf_schedule_abr_task(ospf);
1022 XFREE(MTYPE_OSPF_NETWORK, network);
718e3744 1023}
1024
d62a17ae 1025int ospf_network_set(struct ospf *ospf, struct prefix_ipv4 *p,
1026 struct in_addr area_id, int df)
718e3744 1027{
d62a17ae 1028 struct ospf_network *network;
1029 struct ospf_area *area;
1030 struct route_node *rn;
718e3744 1031
d62a17ae 1032 rn = route_node_get(ospf->networks, (struct prefix *)p);
1033 if (rn->info) {
1034 /* There is already same network statement. */
1035 route_unlock_node(rn);
1036 return 0;
1037 }
718e3744 1038
d62a17ae 1039 rn->info = network = ospf_network_new(area_id);
1040 network->area_id_fmt = df;
1041 area = ospf_area_get(ospf, area_id);
1042 ospf_area_display_format_set(ospf, area, df);
718e3744 1043
d62a17ae 1044 /* Run network config now. */
1045 ospf_network_run((struct prefix *)p, area);
718e3744 1046
d62a17ae 1047 /* Update connected redistribute. */
1048 update_redistributed(ospf, 1); /* interfaces possibly added */
718e3744 1049
d62a17ae 1050 ospf_area_check_free(ospf, area_id);
718e3744 1051
d62a17ae 1052 return 1;
718e3744 1053}
1054
d62a17ae 1055int ospf_network_unset(struct ospf *ospf, struct prefix_ipv4 *p,
1056 struct in_addr area_id)
718e3744 1057{
d62a17ae 1058 struct route_node *rn;
1059 struct ospf_network *network;
1060 struct listnode *node, *nnode;
1061 struct ospf_interface *oi;
718e3744 1062
d62a17ae 1063 rn = route_node_lookup(ospf->networks, (struct prefix *)p);
1064 if (rn == NULL)
1065 return 0;
718e3744 1066
d62a17ae 1067 network = rn->info;
1068 route_unlock_node(rn);
1069 if (!IPV4_ADDR_SAME(&area_id, &network->area_id))
1070 return 0;
718e3744 1071
d62a17ae 1072 ospf_network_free(ospf, rn->info);
1073 rn->info = NULL;
1074 route_unlock_node(rn); /* initial reference */
718e3744 1075
30c0daa4 1076 /* Find interfaces that are not configured already. */
d62a17ae 1077 for (ALL_LIST_ELEMENTS(ospf->oiflist, node, nnode, oi)) {
30c0daa4
JAG
1078
1079 if (oi->type == OSPF_IFTYPE_VIRTUALLINK)
1080 continue;
1081
1082 ospf_network_run_subnet(ospf, oi->connected, NULL, NULL);
d62a17ae 1083 }
953cde65 1084
d62a17ae 1085 /* Update connected redistribute. */
1086 update_redistributed(ospf, 0); /* interfaces possibly removed */
1087 ospf_area_check_free(ospf, area_id);
1088
1089 return 1;
953cde65
JT
1090}
1091
52c62ab8
JAG
1092/* Ensure there's an OSPF instance, as "ip ospf area" enabled OSPF means
1093 * there might not be any 'router ospf' config.
1094 *
1095 * Otherwise, doesn't do anything different to ospf_if_update for now
1096 */
b5a8894d 1097void ospf_interface_area_set(struct ospf *ospf, struct interface *ifp)
953cde65 1098{
b5a8894d
CS
1099 if (!ospf)
1100 return;
d62a17ae 1101
1102 ospf_if_update(ospf, ifp);
1103 /* if_update does a update_redistributed */
1104
1105 return;
953cde65
JT
1106}
1107
b5a8894d 1108void ospf_interface_area_unset(struct ospf *ospf, struct interface *ifp)
953cde65 1109{
d62a17ae 1110 struct route_node *rn_oi;
953cde65 1111
d62a17ae 1112 if (!ospf)
1113 return; /* Ospf not ready yet */
953cde65 1114
d62a17ae 1115 /* Find interfaces that may need to be removed. */
1116 for (rn_oi = route_top(IF_OIFS(ifp)); rn_oi;
1117 rn_oi = route_next(rn_oi)) {
b5a8894d 1118 struct ospf_interface *oi = NULL;
d62a17ae 1119
1120 if ((oi = rn_oi->info) == NULL)
1121 continue;
1122
1123 if (oi->type == OSPF_IFTYPE_VIRTUALLINK)
1124 continue;
1125
1126 ospf_network_run_subnet(ospf, oi->connected, NULL, NULL);
1127 }
953cde65 1128
d62a17ae 1129 /* Update connected redistribute. */
1130 update_redistributed(ospf, 0); /* interfaces possibly removed */
718e3744 1131}
1132
570f7598 1133/* Check whether interface matches given network
1134 * returns: 1, true. 0, false
1135 */
d62a17ae 1136static int ospf_network_match_iface(const struct connected *co,
1137 const struct prefix *net)
570f7598 1138{
d62a17ae 1139 /* new approach: more elegant and conceptually clean */
1140 return prefix_match_network_statement(net, CONNECTED_PREFIX(co));
570f7598 1141}
1142
d62a17ae 1143static void ospf_update_interface_area(struct connected *co,
1144 struct ospf_area *area)
52c62ab8 1145{
d62a17ae 1146 struct ospf_interface *oi = ospf_if_table_lookup(co->ifp, co->address);
1147
1148 /* nothing to be done case */
1149 if (oi && oi->area == area) {
1150 return;
1151 }
1152
1153 if (oi)
1154 ospf_if_free(oi);
1155
1156 add_ospf_interface(co, area);
52c62ab8
JAG
1157}
1158
1159/* Run OSPF for the given subnet, taking into account the following
1160 * possible sources of area configuration, in the given order of preference:
1161 *
1162 * - Whether there is interface+address specific area configuration
1163 * - Whether there is a default area for the interface
1164 * - Whether there is an area given as a parameter.
1165 * - If no specific network prefix/area is supplied, whether there's
1166 * a matching network configured.
1167 */
d62a17ae 1168static void ospf_network_run_subnet(struct ospf *ospf, struct connected *co,
1169 struct prefix *p,
1170 struct ospf_area *given_area)
1171{
1172 struct ospf_interface *oi;
1173 struct ospf_if_params *params;
1174 struct ospf_area *area = NULL;
1175 struct route_node *rn;
1176 int configed = 0;
1177
1178 if (CHECK_FLAG(co->flags, ZEBRA_IFA_SECONDARY))
1179 return;
1180
1181 if (co->address->family != AF_INET)
1182 return;
1183
1184 /* Try determine the appropriate area for this interface + address
1185 * Start by checking interface config
1186 */
1187 params = ospf_lookup_if_params(co->ifp, co->address->u.prefix4);
1188 if (params && OSPF_IF_PARAM_CONFIGURED(params, if_area))
1189 area = ospf_area_get(ospf, params->if_area);
1190 else {
1191 params = IF_DEF_PARAMS(co->ifp);
1192 if (OSPF_IF_PARAM_CONFIGURED(params, if_area))
1193 area = ospf_area_get(ospf, params->if_area);
1194 }
52c62ab8 1195
d62a17ae 1196 /* If we've found an interface and/or addr specific area, then we're
1197 * done
1198 */
1199 if (area) {
1200 ospf_update_interface_area(co, area);
1201 return;
1202 }
718e3744 1203
d62a17ae 1204 /* Otherwise, only remaining possibility is a matching network statement
1205 */
1206 if (p) {
1207 assert(given_area != NULL);
1208
1209 /* Which either was supplied as a parameter.. (e.g. cause a new
1210 * network/area was just added)..
1211 */
1212 if (p->family == co->address->family
1213 && ospf_network_match_iface(co, p))
1214 ospf_update_interface_area(co, given_area);
1215
1216 return;
1217 }
1218
1219 /* Else we have to search the existing network/area config to see
1220 * if any match..
1221 */
1222 for (rn = route_top(ospf->networks); rn; rn = route_next(rn))
1223 if (rn->info != NULL && ospf_network_match_iface(co, &rn->p)) {
1224 struct ospf_network *network =
1225 (struct ospf_network *)rn->info;
1226 area = ospf_area_get(ospf, network->area_id);
1227 ospf_update_interface_area(co, area);
1228 configed = 1;
1229 }
1230
1231 /* If the subnet isn't in any area, deconfigure */
1232 if (!configed && (oi = ospf_if_table_lookup(co->ifp, co->address)))
1233 ospf_if_free(oi);
718e3744 1234}
1235
d62a17ae 1236static void ospf_network_run_interface(struct ospf *ospf, struct interface *ifp,
1237 struct prefix *p,
1238 struct ospf_area *given_area)
718e3744 1239{
d62a17ae 1240 struct listnode *cnode;
1241 struct connected *co;
1242
1243 if (memcmp(ifp->name, "VLINK", 5) == 0)
1244 return;
718e3744 1245
d62a17ae 1246 /* Network prefix without area is nonsensical */
1247 if (p)
1248 assert(given_area != NULL);
1249
1250 /* if interface prefix is match specified prefix,
1251 then create socket and join multicast group. */
1252 for (ALL_LIST_ELEMENTS_RO(ifp->connected, cnode, co))
1253 ospf_network_run_subnet(ospf, co, p, given_area);
718e3744 1254}
1255
d62a17ae 1256static void ospf_network_run(struct prefix *p, struct ospf_area *area)
718e3744 1257{
f4e14fdb 1258 struct vrf *vrf = vrf_lookup_by_id(area->ospf->vrf_id);
d62a17ae 1259 struct interface *ifp;
d62a17ae 1260
1261 /* Schedule Router ID Update. */
1262 if (area->ospf->router_id.s_addr == 0)
1263 ospf_router_id_update(area->ospf);
718e3744 1264
d62a17ae 1265 /* Get target interface. */
451fda4f 1266 FOR_ALL_INTERFACES (vrf, ifp)
d62a17ae 1267 ospf_network_run_interface(area->ospf, ifp, p, area);
718e3744 1268}
1269
d62a17ae 1270void ospf_ls_upd_queue_empty(struct ospf_interface *oi)
1271{
1272 struct route_node *rn;
1273 struct listnode *node, *nnode;
1274 struct list *lst;
1275 struct ospf_lsa *lsa;
1276
1277 /* empty ls update queue */
1278 for (rn = route_top(oi->ls_upd_queue); rn; rn = route_next(rn))
1279 if ((lst = (struct list *)rn->info)) {
1280 for (ALL_LIST_ELEMENTS(lst, node, nnode, lsa))
1281 ospf_lsa_unlock(&lsa); /* oi->ls_upd_queue */
affe9e99 1282 list_delete_and_null(&lst);
d62a17ae 1283 rn->info = NULL;
1284 }
1285
1286 /* remove update event */
1287 if (oi->t_ls_upd_event) {
1288 thread_cancel(oi->t_ls_upd_event);
1289 oi->t_ls_upd_event = NULL;
1290 }
1291}
6b0655a2 1292
d62a17ae 1293void ospf_if_update(struct ospf *ospf, struct interface *ifp)
718e3744 1294{
43b8d1d8 1295
d62a17ae 1296 if (!ospf)
43b8d1d8 1297 return;
b5a8894d
CS
1298
1299 if (IS_DEBUG_OSPF_EVENT)
1300 zlog_debug("%s: interface %s ifp->vrf_id %u ospf vrf %s vrf_id %u router_id %s",
1301 __PRETTY_FUNCTION__, ifp->name, ifp->vrf_id,
1302 ospf_vrf_id_to_name(ospf->vrf_id), ospf->vrf_id,
1303 inet_ntoa(ospf->router_id));
d62a17ae 1304
1305 /* OSPF must be ready. */
1306 if (!ospf_is_ready(ospf))
1307 return;
1308
1309 ospf_network_run_interface(ospf, ifp, NULL, NULL);
1310
1311 /* Update connected redistribute. */
1312 update_redistributed(ospf, 1);
1313}
718e3744 1314
d62a17ae 1315void ospf_remove_vls_through_area(struct ospf *ospf, struct ospf_area *area)
718e3744 1316{
d62a17ae 1317 struct listnode *node, *nnode;
1318 struct ospf_vl_data *vl_data;
718e3744 1319
d62a17ae 1320 for (ALL_LIST_ELEMENTS(ospf->vlinks, node, nnode, vl_data))
1321 if (IPV4_ADDR_SAME(&vl_data->vl_area_id, &area->area_id))
1322 ospf_vl_delete(ospf, vl_data);
1323}
718e3744 1324
718e3744 1325
d62a17ae 1326static const struct message ospf_area_type_msg[] = {
1327 {OSPF_AREA_DEFAULT, "Default"},
1328 {OSPF_AREA_STUB, "Stub"},
1329 {OSPF_AREA_NSSA, "NSSA"},
1330 {0}};
718e3744 1331
d62a17ae 1332static void ospf_area_type_set(struct ospf_area *area, int type)
1333{
1334 struct listnode *node;
1335 struct ospf_interface *oi;
1336
1337 if (area->external_routing == type) {
1338 if (IS_DEBUG_OSPF_EVENT)
1339 zlog_debug("Area[%s]: Types are the same, ignored.",
1340 inet_ntoa(area->area_id));
1341 return;
1342 }
718e3744 1343
d62a17ae 1344 area->external_routing = type;
1345
1346 if (IS_DEBUG_OSPF_EVENT)
1347 zlog_debug("Area[%s]: Configured as %s",
1348 inet_ntoa(area->area_id),
1349 lookup_msg(ospf_area_type_msg, type, NULL));
1350
1351 switch (area->external_routing) {
1352 case OSPF_AREA_DEFAULT:
1353 for (ALL_LIST_ELEMENTS_RO(area->oiflist, node, oi))
1354 if (oi->nbr_self != NULL) {
1355 UNSET_FLAG(oi->nbr_self->options,
1356 OSPF_OPTION_NP);
1357 SET_FLAG(oi->nbr_self->options, OSPF_OPTION_E);
1358 }
1359 break;
1360 case OSPF_AREA_STUB:
1361 for (ALL_LIST_ELEMENTS_RO(area->oiflist, node, oi))
1362 if (oi->nbr_self != NULL) {
1363 if (IS_DEBUG_OSPF_EVENT)
1364 zlog_debug(
1365 "setting options on %s accordingly",
1366 IF_NAME(oi));
1367 UNSET_FLAG(oi->nbr_self->options,
1368 OSPF_OPTION_NP);
1369 UNSET_FLAG(oi->nbr_self->options,
1370 OSPF_OPTION_E);
1371 if (IS_DEBUG_OSPF_EVENT)
1372 zlog_debug("options set on %s: %x",
1373 IF_NAME(oi), OPTIONS(oi));
1374 }
1375 break;
1376 case OSPF_AREA_NSSA:
1377 for (ALL_LIST_ELEMENTS_RO(area->oiflist, node, oi))
1378 if (oi->nbr_self != NULL) {
1379 zlog_debug(
1380 "setting nssa options on %s accordingly",
1381 IF_NAME(oi));
1382 UNSET_FLAG(oi->nbr_self->options,
1383 OSPF_OPTION_E);
1384 SET_FLAG(oi->nbr_self->options, OSPF_OPTION_NP);
1385 zlog_debug("options set on %s: %x", IF_NAME(oi),
1386 OPTIONS(oi));
1387 }
1388 break;
1389 default:
1390 break;
1391 }
1392
1393 ospf_router_lsa_update_area(area);
1394 ospf_schedule_abr_task(area->ospf);
718e3744 1395}
1396
d62a17ae 1397int ospf_area_shortcut_set(struct ospf *ospf, struct ospf_area *area, int mode)
718e3744 1398{
d62a17ae 1399 if (area->shortcut_configured == mode)
1400 return 0;
718e3744 1401
d62a17ae 1402 area->shortcut_configured = mode;
1403 ospf_router_lsa_update_area(area);
1404 ospf_schedule_abr_task(ospf);
718e3744 1405
d62a17ae 1406 ospf_area_check_free(ospf, area->area_id);
718e3744 1407
d62a17ae 1408 return 1;
718e3744 1409}
1410
d62a17ae 1411int ospf_area_shortcut_unset(struct ospf *ospf, struct ospf_area *area)
718e3744 1412{
d62a17ae 1413 area->shortcut_configured = OSPF_SHORTCUT_DEFAULT;
1414 ospf_router_lsa_update_area(area);
1415 ospf_area_check_free(ospf, area->area_id);
1416 ospf_schedule_abr_task(ospf);
718e3744 1417
d62a17ae 1418 return 1;
718e3744 1419}
1420
d62a17ae 1421static int ospf_area_vlink_count(struct ospf *ospf, struct ospf_area *area)
718e3744 1422{
d62a17ae 1423 struct ospf_vl_data *vl;
1424 struct listnode *node;
1425 int count = 0;
718e3744 1426
d62a17ae 1427 for (ALL_LIST_ELEMENTS_RO(ospf->vlinks, node, vl))
1428 if (IPV4_ADDR_SAME(&vl->vl_area_id, &area->area_id))
1429 count++;
718e3744 1430
d62a17ae 1431 return count;
718e3744 1432}
1433
d62a17ae 1434int ospf_area_display_format_set(struct ospf *ospf, struct ospf_area *area,
1435 int df)
86573dcb 1436{
d62a17ae 1437 area->area_id_fmt = df;
86573dcb 1438
d62a17ae 1439 return 1;
86573dcb
QY
1440}
1441
d62a17ae 1442int ospf_area_stub_set(struct ospf *ospf, struct in_addr area_id)
718e3744 1443{
d62a17ae 1444 struct ospf_area *area;
718e3744 1445
d62a17ae 1446 area = ospf_area_get(ospf, area_id);
1447 if (ospf_area_vlink_count(ospf, area))
1448 return 0;
718e3744 1449
d62a17ae 1450 if (area->external_routing != OSPF_AREA_STUB)
1451 ospf_area_type_set(area, OSPF_AREA_STUB);
718e3744 1452
d62a17ae 1453 return 1;
718e3744 1454}
1455
d62a17ae 1456int ospf_area_stub_unset(struct ospf *ospf, struct in_addr area_id)
718e3744 1457{
d62a17ae 1458 struct ospf_area *area;
718e3744 1459
d62a17ae 1460 area = ospf_area_lookup_by_area_id(ospf, area_id);
1461 if (area == NULL)
1462 return 1;
718e3744 1463
d62a17ae 1464 if (area->external_routing == OSPF_AREA_STUB)
1465 ospf_area_type_set(area, OSPF_AREA_DEFAULT);
718e3744 1466
d62a17ae 1467 ospf_area_check_free(ospf, area_id);
718e3744 1468
d62a17ae 1469 return 1;
718e3744 1470}
1471
d62a17ae 1472int ospf_area_no_summary_set(struct ospf *ospf, struct in_addr area_id)
718e3744 1473{
d62a17ae 1474 struct ospf_area *area;
718e3744 1475
d62a17ae 1476 area = ospf_area_get(ospf, area_id);
1477 area->no_summary = 1;
718e3744 1478
d62a17ae 1479 return 1;
718e3744 1480}
1481
d62a17ae 1482int ospf_area_no_summary_unset(struct ospf *ospf, struct in_addr area_id)
718e3744 1483{
d62a17ae 1484 struct ospf_area *area;
718e3744 1485
d62a17ae 1486 area = ospf_area_lookup_by_area_id(ospf, area_id);
1487 if (area == NULL)
1488 return 0;
718e3744 1489
d62a17ae 1490 area->no_summary = 0;
1491 ospf_area_check_free(ospf, area_id);
718e3744 1492
d62a17ae 1493 return 1;
718e3744 1494}
1495
7ef56a73 1496int ospf_area_nssa_no_summary_set(struct ospf *ospf, struct in_addr area_id)
718e3744 1497{
d62a17ae 1498 struct ospf_area *area;
718e3744 1499
d62a17ae 1500 area = ospf_area_get(ospf, area_id);
1501 if (ospf_area_vlink_count(ospf, area))
1502 return 0;
718e3744 1503
d62a17ae 1504 if (area->external_routing != OSPF_AREA_NSSA) {
1505 ospf_area_type_set(area, OSPF_AREA_NSSA);
1506 ospf->anyNSSA++;
7ef56a73 1507 area->NSSATranslatorRole = OSPF_NSSA_ROLE_CANDIDATE;
d62a17ae 1508 }
718e3744 1509
7ef56a73 1510 ospf_area_no_summary_set(ospf, area_id);
084c7844 1511
d62a17ae 1512 return 1;
718e3744 1513}
1514
7ef56a73
CS
1515int ospf_area_nssa_set(struct ospf *ospf, struct in_addr area_id)
1516{
1517 struct ospf_area *area;
1518
1519 area = ospf_area_get(ospf, area_id);
1520 if (ospf_area_vlink_count(ospf, area))
1521 return 0;
1522
1523 if (area->external_routing != OSPF_AREA_NSSA) {
1524 ospf_area_type_set(area, OSPF_AREA_NSSA);
1525 ospf->anyNSSA++;
1526
1527 /* set NSSA area defaults */
1528 area->no_summary = 0;
1529 area->NSSATranslatorRole = OSPF_NSSA_ROLE_CANDIDATE;
1530 area->NSSATranslatorState = OSPF_NSSA_TRANSLATE_DISABLED;
1531 area->NSSATranslatorStabilityInterval =
1532 OSPF_NSSA_TRANS_STABLE_DEFAULT;
1533 }
1534 return 1;
1535}
1536
1537int ospf_area_nssa_unset(struct ospf *ospf, struct in_addr area_id, int argc)
718e3744 1538{
d62a17ae 1539 struct ospf_area *area;
718e3744 1540
d62a17ae 1541 area = ospf_area_lookup_by_area_id(ospf, area_id);
1542 if (area == NULL)
1543 return 0;
718e3744 1544
7ef56a73
CS
1545 /* argc < 5 -> 'no area x nssa' */
1546 if (argc < 5 && area->external_routing == OSPF_AREA_NSSA) {
d62a17ae 1547 ospf->anyNSSA--;
7ef56a73
CS
1548 /* set NSSA area defaults */
1549 area->no_summary = 0;
1550 area->NSSATranslatorRole = OSPF_NSSA_ROLE_CANDIDATE;
1551 area->NSSATranslatorState = OSPF_NSSA_TRANSLATE_DISABLED;
1552 area->NSSATranslatorStabilityInterval =
1553 OSPF_NSSA_TRANS_STABLE_DEFAULT;
d62a17ae 1554 ospf_area_type_set(area, OSPF_AREA_DEFAULT);
7ef56a73
CS
1555 } else {
1556 area->NSSATranslatorRole = OSPF_NSSA_ROLE_CANDIDATE;
d62a17ae 1557 }
718e3744 1558
d62a17ae 1559 ospf_area_check_free(ospf, area_id);
718e3744 1560
d62a17ae 1561 return 1;
718e3744 1562}
1563
d62a17ae 1564int ospf_area_nssa_translator_role_set(struct ospf *ospf,
1565 struct in_addr area_id, int role)
718e3744 1566{
d62a17ae 1567 struct ospf_area *area;
718e3744 1568
d62a17ae 1569 area = ospf_area_lookup_by_area_id(ospf, area_id);
1570 if (area == NULL)
1571 return 0;
718e3744 1572
d62a17ae 1573 area->NSSATranslatorRole = role;
718e3744 1574
d62a17ae 1575 return 1;
718e3744 1576}
1577
075e12f5 1578#if 0
4dadc291 1579/* XXX: unused? Leave for symmetry? */
1580static int
718e3744 1581ospf_area_nssa_translator_role_unset (struct ospf *ospf,
1582 struct in_addr area_id)
1583{
1584 struct ospf_area *area;
1585
68980084 1586 area = ospf_area_lookup_by_area_id (ospf, area_id);
718e3744 1587 if (area == NULL)
1588 return 0;
1589
084c7844 1590 area->NSSATranslatorRole = OSPF_NSSA_ROLE_CANDIDATE;
718e3744 1591
68980084 1592 ospf_area_check_free (ospf, area_id);
718e3744 1593
1594 return 1;
1595}
075e12f5 1596#endif
718e3744 1597
d62a17ae 1598int ospf_area_export_list_set(struct ospf *ospf, struct ospf_area *area,
1599 const char *list_name)
718e3744 1600{
d62a17ae 1601 struct access_list *list;
1602 list = access_list_lookup(AFI_IP, list_name);
718e3744 1603
d62a17ae 1604 EXPORT_LIST(area) = list;
718e3744 1605
d62a17ae 1606 if (EXPORT_NAME(area))
1607 free(EXPORT_NAME(area));
718e3744 1608
d62a17ae 1609 EXPORT_NAME(area) = strdup(list_name);
1610 ospf_schedule_abr_task(ospf);
718e3744 1611
d62a17ae 1612 return 1;
718e3744 1613}
1614
d62a17ae 1615int ospf_area_export_list_unset(struct ospf *ospf, struct ospf_area *area)
718e3744 1616{
1617
d62a17ae 1618 EXPORT_LIST(area) = 0;
718e3744 1619
d62a17ae 1620 if (EXPORT_NAME(area))
1621 free(EXPORT_NAME(area));
718e3744 1622
d62a17ae 1623 EXPORT_NAME(area) = NULL;
718e3744 1624
d62a17ae 1625 ospf_area_check_free(ospf, area->area_id);
718e3744 1626
d62a17ae 1627 ospf_schedule_abr_task(ospf);
1628
1629 return 1;
718e3744 1630}
1631
d62a17ae 1632int ospf_area_import_list_set(struct ospf *ospf, struct ospf_area *area,
1633 const char *name)
718e3744 1634{
d62a17ae 1635 struct access_list *list;
1636 list = access_list_lookup(AFI_IP, name);
718e3744 1637
d62a17ae 1638 IMPORT_LIST(area) = list;
718e3744 1639
d62a17ae 1640 if (IMPORT_NAME(area))
1641 free(IMPORT_NAME(area));
718e3744 1642
d62a17ae 1643 IMPORT_NAME(area) = strdup(name);
1644 ospf_schedule_abr_task(ospf);
718e3744 1645
d62a17ae 1646 return 1;
718e3744 1647}
1648
d62a17ae 1649int ospf_area_import_list_unset(struct ospf *ospf, struct ospf_area *area)
718e3744 1650{
d62a17ae 1651 IMPORT_LIST(area) = 0;
718e3744 1652
d62a17ae 1653 if (IMPORT_NAME(area))
1654 free(IMPORT_NAME(area));
718e3744 1655
d62a17ae 1656 IMPORT_NAME(area) = NULL;
1657 ospf_area_check_free(ospf, area->area_id);
718e3744 1658
d62a17ae 1659 ospf_schedule_abr_task(ospf);
718e3744 1660
d62a17ae 1661 return 1;
718e3744 1662}
1663
d62a17ae 1664int ospf_timers_refresh_set(struct ospf *ospf, int interval)
718e3744 1665{
d62a17ae 1666 int time_left;
718e3744 1667
d62a17ae 1668 if (ospf->lsa_refresh_interval == interval)
1669 return 1;
718e3744 1670
d62a17ae 1671 time_left = ospf->lsa_refresh_interval
1672 - (monotime(NULL) - ospf->lsa_refresher_started);
718e3744 1673
d62a17ae 1674 if (time_left > interval) {
1675 OSPF_TIMER_OFF(ospf->t_lsa_refresher);
1676 thread_add_timer(master, ospf_lsa_refresh_walker, ospf,
1677 interval, &ospf->t_lsa_refresher);
1678 }
1679 ospf->lsa_refresh_interval = interval;
1680
1681 return 1;
718e3744 1682}
1683
d62a17ae 1684int ospf_timers_refresh_unset(struct ospf *ospf)
718e3744 1685{
d62a17ae 1686 int time_left;
718e3744 1687
d62a17ae 1688 time_left = ospf->lsa_refresh_interval
1689 - (monotime(NULL) - ospf->lsa_refresher_started);
718e3744 1690
d62a17ae 1691 if (time_left > OSPF_LSA_REFRESH_INTERVAL_DEFAULT) {
1692 OSPF_TIMER_OFF(ospf->t_lsa_refresher);
1693 ospf->t_lsa_refresher = NULL;
1694 thread_add_timer(master, ospf_lsa_refresh_walker, ospf,
1695 OSPF_LSA_REFRESH_INTERVAL_DEFAULT,
1696 &ospf->t_lsa_refresher);
1697 }
718e3744 1698
d62a17ae 1699 ospf->lsa_refresh_interval = OSPF_LSA_REFRESH_INTERVAL_DEFAULT;
718e3744 1700
d62a17ae 1701 return 1;
718e3744 1702}
1703
6b0655a2 1704
d62a17ae 1705static struct ospf_nbr_nbma *ospf_nbr_nbma_new(void)
718e3744 1706{
d62a17ae 1707 struct ospf_nbr_nbma *nbr_nbma;
718e3744 1708
d62a17ae 1709 nbr_nbma = XCALLOC(MTYPE_OSPF_NEIGHBOR_STATIC,
1710 sizeof(struct ospf_nbr_nbma));
718e3744 1711
d62a17ae 1712 nbr_nbma->priority = OSPF_NEIGHBOR_PRIORITY_DEFAULT;
1713 nbr_nbma->v_poll = OSPF_POLL_INTERVAL_DEFAULT;
718e3744 1714
d62a17ae 1715 return nbr_nbma;
718e3744 1716}
1717
d62a17ae 1718static void ospf_nbr_nbma_free(struct ospf_nbr_nbma *nbr_nbma)
718e3744 1719{
d62a17ae 1720 XFREE(MTYPE_OSPF_NEIGHBOR_STATIC, nbr_nbma);
718e3744 1721}
1722
d62a17ae 1723static void ospf_nbr_nbma_delete(struct ospf *ospf,
1724 struct ospf_nbr_nbma *nbr_nbma)
718e3744 1725{
d62a17ae 1726 struct route_node *rn;
1727 struct prefix_ipv4 p;
718e3744 1728
d62a17ae 1729 p.family = AF_INET;
1730 p.prefix = nbr_nbma->addr;
1731 p.prefixlen = IPV4_MAX_BITLEN;
718e3744 1732
d62a17ae 1733 rn = route_node_lookup(ospf->nbr_nbma, (struct prefix *)&p);
1734 if (rn) {
1735 ospf_nbr_nbma_free(rn->info);
1736 rn->info = NULL;
1737 route_unlock_node(rn);
1738 route_unlock_node(rn);
1739 }
718e3744 1740}
1741
d62a17ae 1742static void ospf_nbr_nbma_down(struct ospf_nbr_nbma *nbr_nbma)
718e3744 1743{
d62a17ae 1744 OSPF_TIMER_OFF(nbr_nbma->t_poll);
718e3744 1745
d62a17ae 1746 if (nbr_nbma->nbr) {
1747 nbr_nbma->nbr->nbr_nbma = NULL;
1748 OSPF_NSM_EVENT_EXECUTE(nbr_nbma->nbr, NSM_KillNbr);
1749 }
718e3744 1750
d62a17ae 1751 if (nbr_nbma->oi)
1752 listnode_delete(nbr_nbma->oi->nbr_nbma, nbr_nbma);
718e3744 1753}
1754
d62a17ae 1755static void ospf_nbr_nbma_add(struct ospf_nbr_nbma *nbr_nbma,
1756 struct ospf_interface *oi)
718e3744 1757{
d62a17ae 1758 struct ospf_neighbor *nbr;
1759 struct route_node *rn;
1760 struct prefix p;
718e3744 1761
d62a17ae 1762 if (oi->type != OSPF_IFTYPE_NBMA)
1763 return;
718e3744 1764
d62a17ae 1765 if (nbr_nbma->nbr != NULL)
1766 return;
718e3744 1767
d62a17ae 1768 if (IPV4_ADDR_SAME(&oi->nbr_self->address.u.prefix4, &nbr_nbma->addr))
1769 return;
718e3744 1770
d62a17ae 1771 nbr_nbma->oi = oi;
1772 listnode_add(oi->nbr_nbma, nbr_nbma);
718e3744 1773
d62a17ae 1774 /* Get neighbor information from table. */
1775 p.family = AF_INET;
1776 p.prefixlen = IPV4_MAX_BITLEN;
1777 p.u.prefix4 = nbr_nbma->addr;
718e3744 1778
d62a17ae 1779 rn = route_node_get(oi->nbrs, (struct prefix *)&p);
1780 if (rn->info) {
1781 nbr = rn->info;
1782 nbr->nbr_nbma = nbr_nbma;
1783 nbr_nbma->nbr = nbr;
718e3744 1784
d62a17ae 1785 route_unlock_node(rn);
1786 } else {
1787 nbr = rn->info = ospf_nbr_new(oi);
1788 nbr->state = NSM_Down;
1789 nbr->src = nbr_nbma->addr;
1790 nbr->nbr_nbma = nbr_nbma;
1791 nbr->priority = nbr_nbma->priority;
1792 nbr->address = p;
718e3744 1793
d62a17ae 1794 nbr_nbma->nbr = nbr;
1795
1796 OSPF_NSM_EVENT_EXECUTE(nbr, NSM_Start);
1797 }
718e3744 1798}
1799
d62a17ae 1800void ospf_nbr_nbma_if_update(struct ospf *ospf, struct ospf_interface *oi)
718e3744 1801{
d62a17ae 1802 struct ospf_nbr_nbma *nbr_nbma;
1803 struct route_node *rn;
1804 struct prefix_ipv4 p;
718e3744 1805
d62a17ae 1806 if (oi->type != OSPF_IFTYPE_NBMA)
1807 return;
718e3744 1808
d62a17ae 1809 for (rn = route_top(ospf->nbr_nbma); rn; rn = route_next(rn))
1810 if ((nbr_nbma = rn->info))
1811 if (nbr_nbma->oi == NULL && nbr_nbma->nbr == NULL) {
1812 p.family = AF_INET;
1813 p.prefix = nbr_nbma->addr;
1814 p.prefixlen = IPV4_MAX_BITLEN;
718e3744 1815
d62a17ae 1816 if (prefix_match(oi->address,
1817 (struct prefix *)&p))
1818 ospf_nbr_nbma_add(nbr_nbma, oi);
1819 }
718e3744 1820}
1821
d62a17ae 1822struct ospf_nbr_nbma *ospf_nbr_nbma_lookup(struct ospf *ospf,
1823 struct in_addr nbr_addr)
718e3744 1824{
d62a17ae 1825 struct route_node *rn;
1826 struct prefix_ipv4 p;
718e3744 1827
d62a17ae 1828 p.family = AF_INET;
1829 p.prefix = nbr_addr;
1830 p.prefixlen = IPV4_MAX_BITLEN;
718e3744 1831
d62a17ae 1832 rn = route_node_lookup(ospf->nbr_nbma, (struct prefix *)&p);
1833 if (rn) {
1834 route_unlock_node(rn);
1835 return rn->info;
1836 }
1837 return NULL;
718e3744 1838}
1839
d62a17ae 1840struct ospf_nbr_nbma *ospf_nbr_nbma_lookup_next(struct ospf *ospf,
1841 struct in_addr *addr, int first)
718e3744 1842{
1843#if 0
1844 struct ospf_nbr_nbma *nbr_nbma;
52dc7ee6 1845 struct listnode *node;
718e3744 1846#endif
1847
d62a17ae 1848 if (ospf == NULL)
1849 return NULL;
718e3744 1850
1851#if 0
1eb8ef25 1852 for (ALL_LIST_ELEMENTS_RO (ospf->nbr_nbma, node, nbr_nbma))
718e3744 1853 {
718e3744 1854 if (first)
1855 {
1856 *addr = nbr_nbma->addr;
1857 return nbr_nbma;
1858 }
1859 else if (ntohl (nbr_nbma->addr.s_addr) > ntohl (addr->s_addr))
1860 {
1861 *addr = nbr_nbma->addr;
1862 return nbr_nbma;
1863 }
1864 }
1865#endif
d62a17ae 1866 return NULL;
718e3744 1867}
1868
d62a17ae 1869int ospf_nbr_nbma_set(struct ospf *ospf, struct in_addr nbr_addr)
718e3744 1870{
d62a17ae 1871 struct ospf_nbr_nbma *nbr_nbma;
1872 struct ospf_interface *oi;
1873 struct prefix_ipv4 p;
1874 struct route_node *rn;
1875 struct listnode *node;
718e3744 1876
d62a17ae 1877 nbr_nbma = ospf_nbr_nbma_lookup(ospf, nbr_addr);
1878 if (nbr_nbma)
1879 return 0;
718e3744 1880
d62a17ae 1881 nbr_nbma = ospf_nbr_nbma_new();
1882 nbr_nbma->addr = nbr_addr;
718e3744 1883
d62a17ae 1884 p.family = AF_INET;
1885 p.prefix = nbr_addr;
1886 p.prefixlen = IPV4_MAX_BITLEN;
718e3744 1887
d62a17ae 1888 rn = route_node_get(ospf->nbr_nbma, (struct prefix *)&p);
1889 if (rn->info)
1890 route_unlock_node(rn);
1891 rn->info = nbr_nbma;
718e3744 1892
d62a17ae 1893 for (ALL_LIST_ELEMENTS_RO(ospf->oiflist, node, oi)) {
1894 if (oi->type == OSPF_IFTYPE_NBMA)
1895 if (prefix_match(oi->address, (struct prefix *)&p)) {
1896 ospf_nbr_nbma_add(nbr_nbma, oi);
1897 break;
1898 }
1899 }
718e3744 1900
d62a17ae 1901 return 1;
718e3744 1902}
1903
d62a17ae 1904int ospf_nbr_nbma_unset(struct ospf *ospf, struct in_addr nbr_addr)
718e3744 1905{
d62a17ae 1906 struct ospf_nbr_nbma *nbr_nbma;
718e3744 1907
d62a17ae 1908 nbr_nbma = ospf_nbr_nbma_lookup(ospf, nbr_addr);
1909 if (nbr_nbma == NULL)
1910 return 0;
718e3744 1911
d62a17ae 1912 ospf_nbr_nbma_down(nbr_nbma);
1913 ospf_nbr_nbma_delete(ospf, nbr_nbma);
718e3744 1914
d62a17ae 1915 return 1;
718e3744 1916}
1917
d62a17ae 1918int ospf_nbr_nbma_priority_set(struct ospf *ospf, struct in_addr nbr_addr,
1919 u_char priority)
718e3744 1920{
d62a17ae 1921 struct ospf_nbr_nbma *nbr_nbma;
718e3744 1922
d62a17ae 1923 nbr_nbma = ospf_nbr_nbma_lookup(ospf, nbr_addr);
1924 if (nbr_nbma == NULL)
1925 return 0;
718e3744 1926
d62a17ae 1927 if (nbr_nbma->priority != priority)
1928 nbr_nbma->priority = priority;
718e3744 1929
d62a17ae 1930 return 1;
718e3744 1931}
1932
d62a17ae 1933int ospf_nbr_nbma_priority_unset(struct ospf *ospf, struct in_addr nbr_addr)
718e3744 1934{
d62a17ae 1935 struct ospf_nbr_nbma *nbr_nbma;
718e3744 1936
d62a17ae 1937 nbr_nbma = ospf_nbr_nbma_lookup(ospf, nbr_addr);
1938 if (nbr_nbma == NULL)
1939 return 0;
718e3744 1940
d62a17ae 1941 if (nbr_nbma != OSPF_NEIGHBOR_PRIORITY_DEFAULT)
1942 nbr_nbma->priority = OSPF_NEIGHBOR_PRIORITY_DEFAULT;
718e3744 1943
d62a17ae 1944 return 1;
718e3744 1945}
1946
d62a17ae 1947int ospf_nbr_nbma_poll_interval_set(struct ospf *ospf, struct in_addr nbr_addr,
1948 unsigned int interval)
718e3744 1949{
d62a17ae 1950 struct ospf_nbr_nbma *nbr_nbma;
718e3744 1951
d62a17ae 1952 nbr_nbma = ospf_nbr_nbma_lookup(ospf, nbr_addr);
1953 if (nbr_nbma == NULL)
1954 return 0;
718e3744 1955
d62a17ae 1956 if (nbr_nbma->v_poll != interval) {
1957 nbr_nbma->v_poll = interval;
1958 if (nbr_nbma->oi && ospf_if_is_up(nbr_nbma->oi)) {
1959 OSPF_TIMER_OFF(nbr_nbma->t_poll);
1960 OSPF_POLL_TIMER_ON(nbr_nbma->t_poll, ospf_poll_timer,
1961 nbr_nbma->v_poll);
1962 }
718e3744 1963 }
718e3744 1964
d62a17ae 1965 return 1;
718e3744 1966}
1967
d62a17ae 1968int ospf_nbr_nbma_poll_interval_unset(struct ospf *ospf, struct in_addr addr)
718e3744 1969{
d62a17ae 1970 struct ospf_nbr_nbma *nbr_nbma;
718e3744 1971
d62a17ae 1972 nbr_nbma = ospf_nbr_nbma_lookup(ospf, addr);
1973 if (nbr_nbma == NULL)
1974 return 0;
718e3744 1975
d62a17ae 1976 if (nbr_nbma->v_poll != OSPF_POLL_INTERVAL_DEFAULT)
1977 nbr_nbma->v_poll = OSPF_POLL_INTERVAL_DEFAULT;
718e3744 1978
d62a17ae 1979 return 1;
718e3744 1980}
1981
d62a17ae 1982void ospf_master_init(struct thread_master *master)
718e3744 1983{
d62a17ae 1984 memset(&ospf_master, 0, sizeof(struct ospf_master));
020709f9 1985
d62a17ae 1986 om = &ospf_master;
1987 om->ospf = list_new();
1988 om->master = master;
020709f9 1989}
b5a8894d
CS
1990
1991/* Link OSPF instance to VRF. */
1992void ospf_vrf_link(struct ospf *ospf, struct vrf *vrf)
1993{
1994 ospf->vrf_id = vrf->vrf_id;
1995 if (vrf->info != (void *)ospf)
1996 vrf->info = (void *)ospf;
1997}
1998
1999/* Unlink OSPF instance from VRF. */
2000void ospf_vrf_unlink(struct ospf *ospf, struct vrf *vrf)
2001{
2002 if (vrf->info == (void *)ospf)
2003 vrf->info = NULL;
2004 ospf->vrf_id = VRF_UNKNOWN;
2005}
2006
2007/* This is hook function for vrf create called as part of vrf_init */
2008static int ospf_vrf_new(struct vrf *vrf)
2009{
2010 if (IS_DEBUG_OSPF_EVENT)
2011 zlog_debug("%s: VRF Created: %s(%d)", __PRETTY_FUNCTION__,
2012 vrf->name, vrf->vrf_id);
2013
2014 return 0;
2015}
2016
2017/* This is hook function for vrf delete call as part of vrf_init */
2018static int ospf_vrf_delete(struct vrf *vrf)
2019{
2020 if (IS_DEBUG_OSPF_EVENT)
2021 zlog_debug("%s: VRF Deletion: %s(%d)", __PRETTY_FUNCTION__,
2022 vrf->name, vrf->vrf_id);
2023
2024 return 0;
2025}
2026
2027/* Enable OSPF VRF instance */
2028static int ospf_vrf_enable(struct vrf *vrf)
2029{
2030 struct ospf *ospf = NULL;
2031 vrf_id_t old_vrf_id = VRF_DEFAULT;
2032
2033 if (IS_DEBUG_OSPF_EVENT)
2034 zlog_debug("%s: VRF %s id %d enabled",
2035 __PRETTY_FUNCTION__, vrf->name, vrf->vrf_id);
2036
2037 ospf = ospf_lookup_by_name(vrf->name);
2038 if (ospf) {
2039 old_vrf_id = ospf->vrf_id;
2040 /* We have instance configured, link to VRF and make it "up". */
2041 ospf_vrf_link(ospf, vrf);
2042 if (IS_DEBUG_OSPF_EVENT)
2043 zlog_debug("%s: ospf linked to vrf %s vrf_id %d (old id %d)",
2044 __PRETTY_FUNCTION__, vrf->name, ospf->vrf_id,
2045 old_vrf_id);
2046
2047 if (old_vrf_id != ospf->vrf_id) {
e7503eab
CS
2048 if (ospfd_privs.change(ZPRIVS_RAISE)) {
2049 zlog_err("ospf_sock_init: could not raise privs, %s",
2050 safe_strerror(errno));
2051 }
2052 if (ospf_bind_vrfdevice(ospf, ospf->fd) < 0)
2053 return 0;
2054 if (ospfd_privs.change(ZPRIVS_LOWER)) {
2055 zlog_err("ospf_sock_init: could not lower privs, %s",
2056 safe_strerror(errno));
2057 }
2058
b5a8894d 2059 ospf->oi_running = 1;
6021c6c0 2060 ospf_zebra_vrf_register(ospf);
b5a8894d
CS
2061 ospf_router_id_update(ospf);
2062 }
2063 }
2064
2065 return 0;
2066}
2067
2068/* Disable OSPF VRF instance */
2069static int ospf_vrf_disable(struct vrf *vrf)
2070{
2071 struct ospf *ospf = NULL;
2072 vrf_id_t old_vrf_id = VRF_UNKNOWN;
2073
2074 if (vrf->vrf_id == VRF_DEFAULT)
2075 return 0;
2076
2077 if (IS_DEBUG_OSPF_EVENT)
2078 zlog_debug("%s: VRF %s id %d disabled.",
2079 __PRETTY_FUNCTION__, vrf->name, vrf->vrf_id);
2080
2081 ospf = ospf_lookup_by_name(vrf->name);
2082 if (ospf) {
2083 old_vrf_id = ospf->vrf_id;
2084
2085 /* We have instance configured, unlink
2086 * from VRF and make it "down".
2087 */
2088 ospf_vrf_unlink(ospf, vrf);
2089 ospf->oi_running = 0;
2090 if (IS_DEBUG_OSPF_EVENT)
2091 zlog_debug("%s: ospf old_vrf_id %d unlinked",
2092 __PRETTY_FUNCTION__, old_vrf_id);
2093 }
2094
2095 /* Note: This is a callback, the VRF will be deleted by the caller. */
2096 return 0;
2097}
2098
2099void ospf_vrf_init(void)
2100{
2101 vrf_init(ospf_vrf_new, ospf_vrf_enable,
2102 ospf_vrf_disable, ospf_vrf_delete);
2103}
2104
2105void ospf_vrf_terminate(void)
2106{
2107 vrf_terminate();
2108}
2109
2110const char *ospf_vrf_id_to_name(vrf_id_t vrf_id)
2111{
2112 struct vrf *vrf = vrf_lookup_by_id(vrf_id);
2113
2114 return vrf ? vrf->name : "NIL";
2115}