blob: bc6ec11962797b3000db9be37c1b630fa5c08aea [file] [log] [blame]
jardineb5d44e2003-12-23 08:09:43 +00001/*
2 * IS-IS Rout(e)ing protocol - isis_dr.c
3 * IS-IS designated router related routines
4 *
5 * Copyright (C) 2001,2002 Sampo Saaristo
6 * Tampere University of Technology
7 * Institute of Communications Engineering
8 *
9 * This program is free software; you can redistribute it and/or modify it
10 * under the terms of the GNU General Public Licenseas published by the Free
11 * Software Foundation; either version 2 of the License, or (at your option)
12 * any later version.
13 *
14 * This program is distributed in the hope that it will be useful,but WITHOUT
15 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
16 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
17 * more details.
18
19 * You should have received a copy of the GNU General Public License along
20 * with this program; if not, write to the Free Software Foundation, Inc.,
21 * 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
22 */
23
24
25#include <zebra.h>
jardineb5d44e2003-12-23 08:09:43 +000026
27#include "log.h"
28#include "hash.h"
29#include "thread.h"
30#include "linklist.h"
31#include "vty.h"
32#include "stream.h"
33#include "if.h"
34
35#include "isisd/dict.h"
36#include "isisd/isis_constants.h"
37#include "isisd/isis_common.h"
38#include "isisd/isis_misc.h"
39#include "isisd/isis_flags.h"
40#include "isisd/isis_circuit.h"
41#include "isisd/isisd.h"
42#include "isisd/isis_adjacency.h"
43#include "isisd/isis_constants.h"
44#include "isisd/isis_pdu.h"
45#include "isisd/isis_tlv.h"
46#include "isisd/isis_lsp.h"
47#include "isisd/isis_dr.h"
48#include "isisd/isis_events.h"
49
hasso1cd80842004-10-07 20:07:40 +000050const char *
hassof390d2c2004-09-10 20:48:21 +000051isis_disflag2string (int disflag)
52{
jardineb5d44e2003-12-23 08:09:43 +000053
hassof390d2c2004-09-10 20:48:21 +000054 switch (disflag)
55 {
jardineb5d44e2003-12-23 08:09:43 +000056 case ISIS_IS_NOT_DIS:
57 return "is not DIS";
58 case ISIS_IS_DIS:
59 return "is DIS";
60 case ISIS_WAS_DIS:
61 return "was DIS";
hassof390d2c2004-09-10 20:48:21 +000062 default:
63 return "unknown DIS state";
64 }
65 return NULL; /* not reached */
jardineb5d44e2003-12-23 08:09:43 +000066}
67
jardineb5d44e2003-12-23 08:09:43 +000068int
69isis_run_dr_l1 (struct thread *thread)
70{
71 struct isis_circuit *circuit;
hassof390d2c2004-09-10 20:48:21 +000072
jardineb5d44e2003-12-23 08:09:43 +000073 circuit = THREAD_ARG (thread);
74 assert (circuit);
75
76 if (circuit->u.bc.run_dr_elect[0])
77 zlog_warn ("isis_run_dr(): run_dr_elect already set for l1");
hassof390d2c2004-09-10 20:48:21 +000078
hassoefc1e722003-12-31 20:33:23 +000079 circuit->u.bc.t_run_dr[0] = NULL;
jardineb5d44e2003-12-23 08:09:43 +000080 circuit->u.bc.run_dr_elect[0] = 1;
hassof390d2c2004-09-10 20:48:21 +000081
jardineb5d44e2003-12-23 08:09:43 +000082 return ISIS_OK;
83}
84
85int
86isis_run_dr_l2 (struct thread *thread)
87{
88 struct isis_circuit *circuit;
hassof390d2c2004-09-10 20:48:21 +000089
jardineb5d44e2003-12-23 08:09:43 +000090 circuit = THREAD_ARG (thread);
91 assert (circuit);
92
93 if (circuit->u.bc.run_dr_elect[1])
94 zlog_warn ("isis_run_dr(): run_dr_elect already set for l2");
hassof390d2c2004-09-10 20:48:21 +000095
96
97 circuit->u.bc.t_run_dr[1] = NULL;
jardineb5d44e2003-12-23 08:09:43 +000098 circuit->u.bc.run_dr_elect[1] = 1;
hassof390d2c2004-09-10 20:48:21 +000099
jardineb5d44e2003-12-23 08:09:43 +0000100 return ISIS_OK;
101}
102
hasso92365882005-01-18 13:53:33 +0000103static int
jardineb5d44e2003-12-23 08:09:43 +0000104isis_check_dr_change (struct isis_adjacency *adj, int level)
105{
106 int i;
107
hassof390d2c2004-09-10 20:48:21 +0000108 if (adj->dis_record[level - 1].dis !=
109 adj->dis_record[(1 * ISIS_LEVELS) + level - 1].dis)
110 /* was there a DIS state transition ? */
jardineb5d44e2003-12-23 08:09:43 +0000111 {
hassof390d2c2004-09-10 20:48:21 +0000112 adj->dischanges[level - 1]++;
jardineb5d44e2003-12-23 08:09:43 +0000113 /* ok rotate the history list through */
hassof390d2c2004-09-10 20:48:21 +0000114 for (i = DIS_RECORDS - 1; i > 0; i--)
jardineb5d44e2003-12-23 08:09:43 +0000115 {
hassof390d2c2004-09-10 20:48:21 +0000116 adj->dis_record[(i * ISIS_LEVELS) + level - 1].dis =
117 adj->dis_record[((i - 1) * ISIS_LEVELS) + level - 1].dis;
118 adj->dis_record[(i * ISIS_LEVELS) + level - 1].last_dis_change =
119 adj->dis_record[((i - 1) * ISIS_LEVELS) + level -
120 1].last_dis_change;
jardineb5d44e2003-12-23 08:09:43 +0000121 }
122 }
123 return ISIS_OK;
124}
125
hassof390d2c2004-09-10 20:48:21 +0000126int
jardineb5d44e2003-12-23 08:09:43 +0000127isis_dr_elect (struct isis_circuit *circuit, int level)
128{
129 struct list *adjdb;
hasso3fdb2dd2005-09-28 18:45:54 +0000130 struct listnode *node;
jardineb5d44e2003-12-23 08:09:43 +0000131 struct isis_adjacency *adj, *adj_dr = NULL;
132 struct list *list = list_new ();
133 u_char own_prio;
134 int biggest_prio = -1;
135 int cmp_res, retval = ISIS_OK;
hassof390d2c2004-09-10 20:48:21 +0000136
Josh Bailey3f045a02012-03-24 08:35:20 -0700137 own_prio = circuit->priority[level - 1];
jardineb5d44e2003-12-23 08:09:43 +0000138 adjdb = circuit->u.bc.adjdb[level - 1];
139
hassof390d2c2004-09-10 20:48:21 +0000140 if (!adjdb)
141 {
142 zlog_warn ("isis_dr_elect() adjdb == NULL");
hassof390d2c2004-09-10 20:48:21 +0000143 list_delete (list);
Josh Bailey3f045a02012-03-24 08:35:20 -0700144 return ISIS_WARNING;
hassof390d2c2004-09-10 20:48:21 +0000145 }
jardineb5d44e2003-12-23 08:09:43 +0000146 isis_adj_build_up_list (adjdb, list);
147
148 /*
149 * Loop the adjacencies and find the one with the biggest priority
150 */
hasso3fdb2dd2005-09-28 18:45:54 +0000151 for (ALL_LIST_ELEMENTS_RO (list, node, adj))
hassof390d2c2004-09-10 20:48:21 +0000152 {
hassof390d2c2004-09-10 20:48:21 +0000153 /* clear flag for show output */
154 adj->dis_record[level - 1].dis = ISIS_IS_NOT_DIS;
155 adj->dis_record[level - 1].last_dis_change = time (NULL);
jardineb5d44e2003-12-23 08:09:43 +0000156
hassof390d2c2004-09-10 20:48:21 +0000157 if (adj->prio[level - 1] > biggest_prio)
158 {
159 biggest_prio = adj->prio[level - 1];
160 adj_dr = adj;
jardineb5d44e2003-12-23 08:09:43 +0000161 }
hassof390d2c2004-09-10 20:48:21 +0000162 else if (adj->prio[level - 1] == biggest_prio)
163 {
164 /*
165 * Comparison of MACs breaks a tie
166 */
167 if (adj_dr)
168 {
169 cmp_res = memcmp (adj_dr->snpa, adj->snpa, ETH_ALEN);
170 if (cmp_res < 0)
171 {
172 adj_dr = adj;
173 }
174 if (cmp_res == 0)
175 zlog_warn
176 ("isis_dr_elect(): multiple adjacencies with same SNPA");
177 }
178 else
179 {
180 adj_dr = adj;
181 }
182 }
jardineb5d44e2003-12-23 08:09:43 +0000183 }
hassof390d2c2004-09-10 20:48:21 +0000184
185 if (!adj_dr)
186 {
187 /*
Josh Bailey3f045a02012-03-24 08:35:20 -0700188 * Could not find the DR - means we are alone. Resign if we were DR.
hassof390d2c2004-09-10 20:48:21 +0000189 */
Josh Bailey3f045a02012-03-24 08:35:20 -0700190 if (circuit->u.bc.is_dr[level - 1])
191 retval = isis_dr_resign (circuit, level);
192 list_delete (list);
193 return retval;
jardineb5d44e2003-12-23 08:09:43 +0000194 }
jardineb5d44e2003-12-23 08:09:43 +0000195
196 /*
197 * Now we have the DR adjacency, compare it to self
198 */
Josh Bailey3f045a02012-03-24 08:35:20 -0700199 if (adj_dr->prio[level - 1] < own_prio ||
200 (adj_dr->prio[level - 1] == own_prio &&
201 memcmp (adj_dr->snpa, circuit->u.bc.snpa, ETH_ALEN) < 0))
hassof390d2c2004-09-10 20:48:21 +0000202 {
Josh Bailey3f045a02012-03-24 08:35:20 -0700203 adj_dr->dis_record[level - 1].dis = ISIS_IS_NOT_DIS;
204 adj_dr->dis_record[level - 1].last_dis_change = time (NULL);
hassod9c427b2004-09-21 14:23:01 +0000205
Josh Bailey3f045a02012-03-24 08:35:20 -0700206 /* rotate the history log */
207 for (ALL_LIST_ELEMENTS_RO (list, node, adj))
208 isis_check_dr_change (adj, level);
hassod9c427b2004-09-21 14:23:01 +0000209
Josh Bailey3f045a02012-03-24 08:35:20 -0700210 /* We are the DR, commence DR */
211 if (circuit->u.bc.is_dr[level - 1] == 0 && listcount (list) > 0)
212 retval = isis_dr_commence (circuit, level);
hassof390d2c2004-09-10 20:48:21 +0000213 }
214 else
215 {
hassof390d2c2004-09-10 20:48:21 +0000216 /* ok we have found the DIS - lets mark the adjacency */
217 /* set flag for show output */
218 adj_dr->dis_record[level - 1].dis = ISIS_IS_DIS;
219 adj_dr->dis_record[level - 1].last_dis_change = time (NULL);
220
221 /* now loop through a second time to check if there has been a DIS change
222 * if yes rotate the history log
jardineb5d44e2003-12-23 08:09:43 +0000223 */
hassof390d2c2004-09-10 20:48:21 +0000224
hasso3fdb2dd2005-09-28 18:45:54 +0000225 for (ALL_LIST_ELEMENTS_RO (list, node, adj))
paul1eb8ef22005-04-07 07:30:20 +0000226 isis_check_dr_change (adj, level);
hassof390d2c2004-09-10 20:48:21 +0000227
228 /*
229 * We are not DR - if we were -> resign
230 */
hassof390d2c2004-09-10 20:48:21 +0000231 if (circuit->u.bc.is_dr[level - 1])
Josh Bailey3f045a02012-03-24 08:35:20 -0700232 retval = isis_dr_resign (circuit, level);
jardineb5d44e2003-12-23 08:09:43 +0000233 }
Josh Bailey3f045a02012-03-24 08:35:20 -0700234 list_delete (list);
jardineb5d44e2003-12-23 08:09:43 +0000235 return retval;
236}
237
hassof390d2c2004-09-10 20:48:21 +0000238int
jardineb5d44e2003-12-23 08:09:43 +0000239isis_dr_resign (struct isis_circuit *circuit, int level)
240{
241 u_char id[ISIS_SYS_ID_LEN + 2];
hassof390d2c2004-09-10 20:48:21 +0000242
hasso529d65b2004-12-24 00:14:50 +0000243 zlog_debug ("isis_dr_resign l%d", level);
jardineb5d44e2003-12-23 08:09:43 +0000244
245 circuit->u.bc.is_dr[level - 1] = 0;
hassof390d2c2004-09-10 20:48:21 +0000246 circuit->u.bc.run_dr_elect[level - 1] = 0;
247 THREAD_TIMER_OFF (circuit->u.bc.t_run_dr[level - 1]);
248 THREAD_TIMER_OFF (circuit->u.bc.t_refresh_pseudo_lsp[level - 1]);
Josh Bailey3f045a02012-03-24 08:35:20 -0700249 circuit->lsp_regenerate_pending[level - 1] = 0;
hassof390d2c2004-09-10 20:48:21 +0000250
jardineb5d44e2003-12-23 08:09:43 +0000251 memcpy (id, isis->sysid, ISIS_SYS_ID_LEN);
hassof390d2c2004-09-10 20:48:21 +0000252 LSP_PSEUDO_ID (id) = circuit->circuit_id;
253 LSP_FRAGMENT (id) = 0;
Josh Bailey3f045a02012-03-24 08:35:20 -0700254 lsp_purge_pseudo (id, circuit, level);
jardineb5d44e2003-12-23 08:09:43 +0000255
hassof390d2c2004-09-10 20:48:21 +0000256 if (level == 1)
257 {
258 memset (circuit->u.bc.l1_desig_is, 0, ISIS_SYS_ID_LEN + 1);
jardineb5d44e2003-12-23 08:09:43 +0000259
hassof390d2c2004-09-10 20:48:21 +0000260 THREAD_TIMER_OFF (circuit->t_send_csnp[0]);
jardineb5d44e2003-12-23 08:09:43 +0000261
hassof390d2c2004-09-10 20:48:21 +0000262 THREAD_TIMER_ON (master, circuit->u.bc.t_run_dr[0], isis_run_dr_l1,
hassoe7fe02f2005-05-26 11:30:38 +0000263 circuit, 2 * circuit->hello_interval[0]);
hassof390d2c2004-09-10 20:48:21 +0000264
265 THREAD_TIMER_ON (master, circuit->t_send_psnp[0], send_l1_psnp, circuit,
266 isis_jitter (circuit->psnp_interval[level - 1],
267 PSNP_JITTER));
268 }
269 else
270 {
271 memset (circuit->u.bc.l2_desig_is, 0, ISIS_SYS_ID_LEN + 1);
272
273 THREAD_TIMER_OFF (circuit->t_send_csnp[1]);
274
275 THREAD_TIMER_ON (master, circuit->u.bc.t_run_dr[1], isis_run_dr_l2,
276 circuit, 2 * circuit->hello_interval[1]);
277
278 THREAD_TIMER_ON (master, circuit->t_send_psnp[1], send_l2_psnp, circuit,
279 isis_jitter (circuit->psnp_interval[level - 1],
280 PSNP_JITTER));
281 }
282
jardineb5d44e2003-12-23 08:09:43 +0000283 thread_add_event (master, isis_event_dis_status_change, circuit, 0);
284
285 return ISIS_OK;
286}
287
hassof390d2c2004-09-10 20:48:21 +0000288int
jardineb5d44e2003-12-23 08:09:43 +0000289isis_dr_commence (struct isis_circuit *circuit, int level)
290{
291 u_char old_dr[ISIS_SYS_ID_LEN + 2];
hassof390d2c2004-09-10 20:48:21 +0000292
hassoc89c05d2005-09-04 21:36:36 +0000293 if (isis->debugs & DEBUG_EVENTS)
294 zlog_debug ("isis_dr_commence l%d", level);
jardineb5d44e2003-12-23 08:09:43 +0000295
296 /* Lets keep a pause in DR election */
297 circuit->u.bc.run_dr_elect[level - 1] = 0;
hassof390d2c2004-09-10 20:48:21 +0000298 if (level == 1)
299 THREAD_TIMER_ON (master, circuit->u.bc.t_run_dr[0], isis_run_dr_l1,
hassoa211d652004-09-20 14:55:29 +0000300 circuit, 2 * circuit->hello_interval[0]);
hassof390d2c2004-09-10 20:48:21 +0000301 else
302 THREAD_TIMER_ON (master, circuit->u.bc.t_run_dr[1], isis_run_dr_l2,
hassoa211d652004-09-20 14:55:29 +0000303 circuit, 2 * circuit->hello_interval[1]);
jardineb5d44e2003-12-23 08:09:43 +0000304 circuit->u.bc.is_dr[level - 1] = 1;
305
hassof390d2c2004-09-10 20:48:21 +0000306 if (level == 1)
307 {
308 memcpy (old_dr, circuit->u.bc.l1_desig_is, ISIS_SYS_ID_LEN + 1);
309 LSP_FRAGMENT (old_dr) = 0;
310 if (LSP_PSEUDO_ID (old_dr))
311 {
312 /* there was a dr elected, purge its LSPs from the db */
Josh Bailey3f045a02012-03-24 08:35:20 -0700313 lsp_purge_pseudo (old_dr, circuit, level);
hassof390d2c2004-09-10 20:48:21 +0000314 }
315 memcpy (circuit->u.bc.l1_desig_is, isis->sysid, ISIS_SYS_ID_LEN);
316 *(circuit->u.bc.l1_desig_is + ISIS_SYS_ID_LEN) = circuit->circuit_id;
jardineb5d44e2003-12-23 08:09:43 +0000317
hassof390d2c2004-09-10 20:48:21 +0000318 assert (circuit->circuit_id); /* must be non-zero */
319 /* if (circuit->t_send_l1_psnp)
320 thread_cancel (circuit->t_send_l1_psnp); */
Josh Bailey3f045a02012-03-24 08:35:20 -0700321 lsp_generate_pseudo (circuit, 1);
jardineb5d44e2003-12-23 08:09:43 +0000322
hassof390d2c2004-09-10 20:48:21 +0000323 THREAD_TIMER_OFF (circuit->u.bc.t_run_dr[0]);
324 THREAD_TIMER_ON (master, circuit->u.bc.t_run_dr[0], isis_run_dr_l1,
325 circuit, 2 * circuit->hello_interval[0]);
326
327 THREAD_TIMER_ON (master, circuit->t_send_csnp[0], send_l1_csnp, circuit,
328 isis_jitter (circuit->csnp_interval[level - 1],
329 CSNP_JITTER));
330
331 }
332 else
333 {
334 memcpy (old_dr, circuit->u.bc.l2_desig_is, ISIS_SYS_ID_LEN + 1);
335 LSP_FRAGMENT (old_dr) = 0;
336 if (LSP_PSEUDO_ID (old_dr))
337 {
338 /* there was a dr elected, purge its LSPs from the db */
Josh Bailey3f045a02012-03-24 08:35:20 -0700339 lsp_purge_pseudo (old_dr, circuit, level);
hassof390d2c2004-09-10 20:48:21 +0000340 }
341 memcpy (circuit->u.bc.l2_desig_is, isis->sysid, ISIS_SYS_ID_LEN);
342 *(circuit->u.bc.l2_desig_is + ISIS_SYS_ID_LEN) = circuit->circuit_id;
343
344 assert (circuit->circuit_id); /* must be non-zero */
345 /* if (circuit->t_send_l1_psnp)
346 thread_cancel (circuit->t_send_l1_psnp); */
Josh Bailey3f045a02012-03-24 08:35:20 -0700347 lsp_generate_pseudo (circuit, 2);
hassof390d2c2004-09-10 20:48:21 +0000348
349 THREAD_TIMER_OFF (circuit->u.bc.t_run_dr[1]);
350 THREAD_TIMER_ON (master, circuit->u.bc.t_run_dr[1], isis_run_dr_l2,
351 circuit, 2 * circuit->hello_interval[1]);
352
353 THREAD_TIMER_ON (master, circuit->t_send_csnp[1], send_l2_csnp, circuit,
354 isis_jitter (circuit->csnp_interval[level - 1],
355 CSNP_JITTER));
356 }
jardineb5d44e2003-12-23 08:09:43 +0000357
358 thread_add_event (master, isis_event_dis_status_change, circuit, 0);
hassof390d2c2004-09-10 20:48:21 +0000359
jardineb5d44e2003-12-23 08:09:43 +0000360 return ISIS_OK;
361}