| 
									
										
										
										
											1998-05-20 11:54:33 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *	BIRD -- Protocols | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |  *	(c) 1998--2000 Martin Mares <mj@ucw.cz> | 
					
						
							| 
									
										
										
										
											1998-05-20 11:54:33 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  *	Can be freely distributed and used under the terms of the GNU GPL. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-12 21:01:38 +00:00
										 |  |  | #undef LOCAL_DEBUG
 | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-05-20 11:54:33 +00:00
										 |  |  | #include "nest/bird.h"
 | 
					
						
							|  |  |  | #include "nest/protocol.h"
 | 
					
						
							|  |  |  | #include "lib/resource.h"
 | 
					
						
							|  |  |  | #include "lib/lists.h"
 | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | #include "lib/event.h"
 | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  | #include "lib/string.h"
 | 
					
						
							| 
									
										
										
										
											1998-11-27 19:35:10 +00:00
										 |  |  | #include "conf/conf.h"
 | 
					
						
							| 
									
										
										
										
											1998-10-17 11:05:18 +00:00
										 |  |  | #include "nest/route.h"
 | 
					
						
							|  |  |  | #include "nest/iface.h"
 | 
					
						
							| 
									
										
										
										
											1999-11-25 15:35:30 +00:00
										 |  |  | #include "nest/cli.h"
 | 
					
						
							| 
									
										
										
										
											1999-03-17 14:31:26 +00:00
										 |  |  | #include "filter/filter.h"
 | 
					
						
							| 
									
										
										
										
											1998-05-20 11:54:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-02 22:20:40 +02:00
										 |  |  | pool *proto_pool; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-04-01 10:19:47 +00:00
										 |  |  | static list protocol_list; | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  | static list proto_list; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-07 21:50:21 +00:00
										 |  |  | #define PD(pr, msg, args...) do { if (pr->debug & D_STATES) { log(L_TRACE "%s: " msg, pr->name , ## args); } } while(0)
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  | list active_proto_list; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | static list inactive_proto_list; | 
					
						
							|  |  |  | static list initial_proto_list; | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  | static list flush_proto_list; | 
					
						
							| 
									
										
										
										
											2011-04-01 13:54:39 +02:00
										 |  |  | static struct proto *initial_device_proto; | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static event *proto_flush_event; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | static timer *proto_shutdown_timer; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | static timer *gr_wait_timer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define GRS_NONE	0
 | 
					
						
							|  |  |  | #define GRS_INIT	1
 | 
					
						
							|  |  |  | #define GRS_ACTIVE	2
 | 
					
						
							|  |  |  | #define GRS_DONE	3
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int graceful_restart_state; | 
					
						
							|  |  |  | static u32 graceful_restart_locks; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static char *p_states[] = { "DOWN", "START", "UP", "STOP" }; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | static char *c_states[] = { "HUNGRY", "???", "HAPPY", "FLUSHING" }; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  | static void proto_flush_loop(void *); | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | static void proto_shutdown_loop(struct timer *); | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | static void proto_rethink_goal(struct proto *p); | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | static void proto_want_export_up(struct proto *p); | 
					
						
							|  |  |  | static void proto_fell_down(struct proto *p); | 
					
						
							| 
									
										
										
										
											2000-03-07 21:50:21 +00:00
										 |  |  | static char *proto_state_name(struct proto *p); | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-03-03 19:33:54 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  | proto_relink(struct proto *p) | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-21 14:34:53 +01:00
										 |  |  |   list *l = NULL; | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   switch (p->core_state) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  |     case FS_HUNGRY: | 
					
						
							|  |  |  |       l = &inactive_proto_list; | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  |     case FS_HAPPY: | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  |       l = &active_proto_list; | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     case FS_FLUSHING: | 
					
						
							|  |  |  |       l = &flush_proto_list; | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  |       ASSERT(0); | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   rem_node(&p->n); | 
					
						
							|  |  |  |   add_tail(l, &p->n); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_log_state_change(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (p->debug & D_STATES) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       char *name = proto_state_name(p); | 
					
						
							|  |  |  |       if (name != p->last_state_name_announced) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  p->last_state_name_announced = name; | 
					
						
							|  |  |  | 	  PD(p, "State changed to %s", proto_state_name(p)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     p->last_state_name_announced = NULL; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1998-05-20 11:54:33 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_new - create a new protocol instance | 
					
						
							|  |  |  |  * @c: protocol configuration | 
					
						
							|  |  |  |  * @size: size of protocol data structure (each protocol instance is represented by | 
					
						
							|  |  |  |  * a structure starting with generic part [struct &proto] and continued | 
					
						
							|  |  |  |  * with data specific to the protocol) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * When a new configuration has been read in, the core code starts | 
					
						
							| 
									
										
										
										
											2000-06-07 12:29:08 +00:00
										 |  |  |  * initializing all the protocol instances configured by calling their | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  * init() hooks with the corresponding instance configuration. The initialization | 
					
						
							|  |  |  |  * code of the protocol is expected to create a new instance according to the | 
					
						
							|  |  |  |  * configuration by calling this function and then modifying the default settings | 
					
						
							|  |  |  |  * to values wanted by the protocol. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  | void * | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | proto_new(struct proto_config *c, unsigned size) | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |   struct protocol *pr = c->protocol; | 
					
						
							| 
									
										
										
										
											1999-03-04 11:39:24 +00:00
										 |  |  |   struct proto *p = mb_allocz(proto_pool, size); | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |   p->cf = c; | 
					
						
							|  |  |  |   p->debug = c->debug; | 
					
						
							| 
									
										
										
										
											2010-01-03 12:17:52 +01:00
										 |  |  |   p->mrtdump = c->mrtdump; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |   p->name = c->name; | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |   p->preference = c->preference; | 
					
						
							|  |  |  |   p->disabled = c->disabled; | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  |   p->proto = pr; | 
					
						
							| 
									
										
										
										
											1999-08-03 19:31:54 +00:00
										 |  |  |   p->table = c->table->table; | 
					
						
							| 
									
										
										
										
											2000-03-01 14:49:07 +00:00
										 |  |  |   p->hash_key = random_u32(); | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |   c->proto = p; | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  |   return p; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | proto_init_instance(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											1999-03-29 20:14:33 +00:00
										 |  |  |   /* Here we cannot use p->cf->name since it won't survive reconfiguration */ | 
					
						
							|  |  |  |   p->pool = rp_new(proto_pool, p->proto->name); | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  |   p->attn = ev_new(p->pool); | 
					
						
							|  |  |  |   p->attn->data = p; | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   if (graceful_restart_state == GRS_INIT) | 
					
						
							|  |  |  |     p->gr_recovery = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   if (! p->proto->multitable) | 
					
						
							|  |  |  |     rt_lock_table(p->table); | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | extern pool *rt_table_pool; | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_add_announce_hook - connect protocol to a routing table | 
					
						
							|  |  |  |  * @p: protocol instance | 
					
						
							|  |  |  |  * @t: routing table to connect to | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |  * @stats: per-table protocol statistics | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  |  * This function creates a connection between the protocol instance @p and the | 
					
						
							|  |  |  |  * routing table @t, making the protocol hear all changes in the table. | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  |  * The announce hook is linked in the protocol ahook list. Announce hooks are | 
					
						
							|  |  |  |  * allocated from the routing table resource pool and when protocol accepts | 
					
						
							|  |  |  |  * routes also in the table ahook list. The are linked to the table ahook list | 
					
						
							|  |  |  |  * and unlinked from it depending on export_state (in proto_want_export_up() and | 
					
						
							|  |  |  |  * proto_want_export_down()) and they are automatically freed after the protocol | 
					
						
							|  |  |  |  * is flushed (in proto_fell_down()). | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  |  * Unless you want to listen to multiple routing tables (as the Pipe protocol | 
					
						
							|  |  |  |  * does), you needn't to worry about this function since the connection to the | 
					
						
							|  |  |  |  * protocol's primary routing table is initialized automatically by the core | 
					
						
							|  |  |  |  * code. | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  | struct announce_hook * | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | proto_add_announce_hook(struct proto *p, struct rtable *t, struct proto_stats *stats) | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   struct announce_hook *h; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DBG("Connecting protocol %s to table %s\n", p->name, t->name); | 
					
						
							| 
									
										
										
										
											2000-03-07 21:50:21 +00:00
										 |  |  |   PD(p, "Connected to table %s", t->name); | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   h = mb_allocz(rt_table_pool, sizeof(struct announce_hook)); | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  |   h->table = t; | 
					
						
							|  |  |  |   h->proto = p; | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   h->stats = stats; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  |   h->next = p->ahooks; | 
					
						
							|  |  |  |   p->ahooks = h; | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  |   if (p->rt_notify && (p->export_state != ES_DOWN)) | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |     add_tail(&t->hooks, &h->n); | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  |   return h; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_find_announce_hook - find announce hooks | 
					
						
							|  |  |  |  * @p: protocol instance | 
					
						
							|  |  |  |  * @t: routing table | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Returns pointer to announce hook or NULL | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct announce_hook * | 
					
						
							|  |  |  | proto_find_announce_hook(struct proto *p, struct rtable *t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct announce_hook *a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (a = p->ahooks; a; a = a->next) | 
					
						
							|  |  |  |     if (a->table == t) | 
					
						
							|  |  |  |       return a; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | static void | 
					
						
							|  |  |  | proto_link_ahooks(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct announce_hook *h; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (p->rt_notify) | 
					
						
							|  |  |  |     for(h=p->ahooks; h; h=h->next) | 
					
						
							|  |  |  |       add_tail(&h->table->hooks, &h->n); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | proto_unlink_ahooks(struct proto *p) | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   struct announce_hook *h; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   if (p->rt_notify) | 
					
						
							|  |  |  |     for(h=p->ahooks; h; h=h->next) | 
					
						
							|  |  |  |       rem_node(&h->n); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_free_ahooks(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct announce_hook *h, *hn; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for(h = p->ahooks; h; h = hn) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     hn = h->next; | 
					
						
							|  |  |  |     mb_free(h); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  |   p->ahooks = NULL; | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   p->main_ahook = NULL; | 
					
						
							| 
									
										
										
										
											1999-05-17 20:14:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-14 16:25:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_config_new - create a new protocol configuration | 
					
						
							|  |  |  |  * @pr: protocol the configuration will belong to | 
					
						
							| 
									
										
										
										
											2011-11-07 00:31:23 +01:00
										 |  |  |  * @class: SYM_PROTO or SYM_TEMPLATE | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Whenever the configuration file says that a new instance | 
					
						
							|  |  |  |  * of a routing protocol should be created, the parser calls | 
					
						
							|  |  |  |  * proto_config_new() to create a configuration entry for this | 
					
						
							|  |  |  |  * instance (a structure staring with the &proto_config header | 
					
						
							|  |  |  |  * containing all the generic items followed by protocol-specific | 
					
						
							|  |  |  |  * ones). Also, the configuration entry gets added to the list | 
					
						
							|  |  |  |  * of protocol instances kept in the configuration. | 
					
						
							| 
									
										
										
										
											2011-11-07 00:31:23 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The function is also used to create protocol templates (when class | 
					
						
							|  |  |  |  * SYM_TEMPLATE is specified), the only difference is that templates | 
					
						
							|  |  |  |  * are not added to the list of protocol instances and therefore not | 
					
						
							|  |  |  |  * initialized during protos_commit()). | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | void * | 
					
						
							| 
									
										
										
										
											2015-02-21 21:08:23 +01:00
										 |  |  | proto_config_new(struct protocol *pr, int class) | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2015-02-21 21:08:23 +01:00
										 |  |  |   struct proto_config *c = cfg_allocz(pr->config_size); | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-07 00:31:23 +01:00
										 |  |  |   if (class == SYM_PROTO) | 
					
						
							|  |  |  |     add_tail(&new_config->protos, &c->n); | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |   c->global = new_config; | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |   c->protocol = pr; | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |   c->name = pr->name; | 
					
						
							| 
									
										
										
										
											2012-01-24 11:31:00 +01:00
										 |  |  |   c->preference = pr->preference; | 
					
						
							| 
									
										
										
										
											2011-11-07 00:31:23 +01:00
										 |  |  |   c->class = class; | 
					
						
							| 
									
										
										
										
											1999-04-05 20:15:31 +00:00
										 |  |  |   c->out_filter = FILTER_REJECT; | 
					
						
							| 
									
										
										
										
											1999-08-03 19:31:54 +00:00
										 |  |  |   c->table = c->global->master_rtc; | 
					
						
							| 
									
										
										
										
											2000-03-07 21:50:21 +00:00
										 |  |  |   c->debug = new_config->proto_default_debug; | 
					
						
							| 
									
										
										
										
											2010-01-03 12:17:52 +01:00
										 |  |  |   c->mrtdump = new_config->proto_default_mrtdump; | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |   return c; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-11-07 00:31:23 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_copy_config - copy a protocol configuration | 
					
						
							|  |  |  |  * @dest: destination protocol configuration | 
					
						
							|  |  |  |  * @src: source protocol configuration | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Whenever a new instance of a routing protocol is created from the | 
					
						
							|  |  |  |  * template, proto_copy_config() is called to copy a content of | 
					
						
							|  |  |  |  * the source protocol configuration to the new protocol configuration. | 
					
						
							|  |  |  |  * Name, class and a node in protos list of @dest are kept intact. | 
					
						
							|  |  |  |  * copy_config() protocol hook is used to copy protocol-specific data. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | proto_copy_config(struct proto_config *dest, struct proto_config *src) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   node old_node; | 
					
						
							|  |  |  |   int old_class; | 
					
						
							|  |  |  |   char *old_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (dest->protocol != src->protocol) | 
					
						
							|  |  |  |     cf_error("Can't copy configuration from a different protocol type"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (dest->protocol->copy_config == NULL) | 
					
						
							|  |  |  |     cf_error("Inheriting configuration for %s is not supported", src->protocol->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DBG("Copying configuration from %s to %s\n", src->name, dest->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* 
 | 
					
						
							|  |  |  |    * Copy struct proto_config here. Keep original node, class and name. | 
					
						
							|  |  |  |    * protocol-specific config copy is handled by protocol copy_config() hook | 
					
						
							|  |  |  |    */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   old_node = dest->n; | 
					
						
							|  |  |  |   old_class = dest->class; | 
					
						
							|  |  |  |   old_name = dest->name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   memcpy(dest, src, sizeof(struct proto_config)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dest->n = old_node; | 
					
						
							|  |  |  |   dest->class = old_class; | 
					
						
							|  |  |  |   dest->name = old_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   dest->protocol->copy_config(dest, src); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * protos_preconfig - pre-configuration processing | 
					
						
							|  |  |  |  * @c: new configuration | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function calls the preconfig() hooks of all routing | 
					
						
							|  |  |  |  * protocols available to prepare them for reading of the new | 
					
						
							|  |  |  |  * configuration. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											1998-05-20 11:54:33 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | protos_preconfig(struct config *c) | 
					
						
							| 
									
										
										
										
											1998-05-20 11:54:33 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  |   struct protocol *p; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-06 13:44:45 +00:00
										 |  |  |   init_list(&c->protos); | 
					
						
							| 
									
										
										
										
											2000-03-12 21:01:38 +00:00
										 |  |  |   DBG("Protocol preconfig:"); | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  |   WALK_LIST(p, protocol_list) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2000-03-12 21:01:38 +00:00
										 |  |  |       DBG(" %s", p->name); | 
					
						
							| 
									
										
										
										
											1999-03-26 21:37:29 +00:00
										 |  |  |       p->name_counter = 0; | 
					
						
							| 
									
										
										
										
											1998-10-18 12:26:02 +00:00
										 |  |  |       if (p->preconfig) | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | 	p->preconfig(p, c); | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2000-03-12 21:01:38 +00:00
										 |  |  |   DBG("\n"); | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * protos_postconfig - post-configuration processing | 
					
						
							|  |  |  |  * @c: new configuration | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function calls the postconfig() hooks of all protocol | 
					
						
							| 
									
										
										
										
											2011-11-07 00:31:23 +01:00
										 |  |  |  * instances specified in configuration @c. The hooks are not | 
					
						
							|  |  |  |  * called for protocol templates. | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | protos_postconfig(struct config *c) | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |   struct proto_config *x; | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  |   struct protocol *p; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-03-12 21:01:38 +00:00
										 |  |  |   DBG("Protocol postconfig:"); | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |   WALK_LIST(x, c->protos) | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2000-03-12 21:01:38 +00:00
										 |  |  |       DBG(" %s", x->name); | 
					
						
							| 
									
										
										
										
											2013-01-10 13:07:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |       p = x->protocol; | 
					
						
							| 
									
										
										
										
											1998-10-18 12:26:02 +00:00
										 |  |  |       if (p->postconfig) | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | 	p->postconfig(x); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2000-03-12 21:01:38 +00:00
										 |  |  |   DBG("\n"); | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-01 13:54:39 +02:00
										 |  |  | extern struct protocol proto_unix_iface; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | static struct proto * | 
					
						
							|  |  |  | proto_init(struct proto_config *c) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct protocol *p = c->protocol; | 
					
						
							|  |  |  |   struct proto *q = p->init(c); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   q->proto_state = PS_DOWN; | 
					
						
							|  |  |  |   q->core_state = FS_HUNGRY; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   q->export_state = ES_DOWN; | 
					
						
							| 
									
										
										
										
											2013-11-24 22:22:24 +01:00
										 |  |  |   q->last_state_change = now; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  |   add_tail(&initial_proto_list, &q->n); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-04-01 13:54:39 +02:00
										 |  |  |   if (p == &proto_unix_iface) | 
					
						
							|  |  |  |     initial_device_proto = q; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  |   add_tail(&proto_list, &q->glob_node); | 
					
						
							| 
									
										
										
										
											2000-05-06 21:42:19 +00:00
										 |  |  |   PD(q, "Initializing%s", q->disabled ? " [disabled]" : ""); | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |   return q; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | int proto_reconfig_type;  /* Hack to propagate type info to pipe reconfigure hook */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 19:19:09 +01:00
										 |  |  | static int | 
					
						
							|  |  |  | proto_reconfigure(struct proto *p, struct proto_config *oc, struct proto_config *nc, int type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* If the protocol is DOWN, we just restart it */ | 
					
						
							|  |  |  |   if (p->proto_state == PS_DOWN) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* If there is a too big change in core attributes, ... */ | 
					
						
							|  |  |  |   if ((nc->protocol != oc->protocol) || | 
					
						
							| 
									
										
										
										
											2011-06-20 20:35:59 +02:00
										 |  |  |       (nc->disabled != p->disabled) || | 
					
						
							| 
									
										
										
										
											2012-12-27 12:56:23 +01:00
										 |  |  |       (nc->table->table != oc->table->table)) | 
					
						
							| 
									
										
										
										
											2010-02-06 19:19:09 +01:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p->debug = nc->debug; | 
					
						
							|  |  |  |   p->mrtdump = nc->mrtdump; | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   proto_reconfig_type = type; | 
					
						
							| 
									
										
										
										
											2010-02-06 19:19:09 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Execute protocol specific reconfigure hook */ | 
					
						
							|  |  |  |   if (! (p->proto->reconfigure && p->proto->reconfigure(p, nc))) | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DBG("\t%s: same\n", oc->name); | 
					
						
							|  |  |  |   PD(p, "Reconfigured"); | 
					
						
							|  |  |  |   p->cf = nc; | 
					
						
							|  |  |  |   p->name = nc->name; | 
					
						
							| 
									
										
										
										
											2010-02-08 12:42:09 +01:00
										 |  |  |   p->preference = nc->preference; | 
					
						
							| 
									
										
										
										
											2010-02-06 19:19:09 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Multitable protocols handle rest in their reconfigure hooks */ | 
					
						
							|  |  |  |   if (p->proto->multitable) | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-24 23:39:57 +02:00
										 |  |  |   /* Update filters and limits in the main announce hook
 | 
					
						
							|  |  |  |      Note that this also resets limit state */ | 
					
						
							| 
									
										
										
										
											2014-04-27 00:46:32 +02:00
										 |  |  |   if (p->main_ahook) | 
					
						
							|  |  |  |     {   | 
					
						
							|  |  |  |       struct announce_hook *ah = p->main_ahook; | 
					
						
							| 
									
										
										
										
											2013-09-30 14:07:34 +02:00
										 |  |  |       ah->in_filter = nc->in_filter; | 
					
						
							|  |  |  |       ah->out_filter = nc->out_filter; | 
					
						
							|  |  |  |       ah->rx_limit = nc->rx_limit; | 
					
						
							|  |  |  |       ah->in_limit = nc->in_limit; | 
					
						
							|  |  |  |       ah->out_limit = nc->out_limit; | 
					
						
							|  |  |  |       ah->in_keep_filtered = nc->in_keep_filtered; | 
					
						
							| 
									
										
										
										
											2014-04-27 00:46:32 +02:00
										 |  |  |       proto_verify_limits(ah); | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Update routes when filters changed. If the protocol in not UP,
 | 
					
						
							|  |  |  |      it has no routes and we can ignore such changes */ | 
					
						
							|  |  |  |   if ((p->proto_state != PS_UP) || (type == RECONFIG_SOFT)) | 
					
						
							|  |  |  |     return 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   int import_changed = ! filter_same(nc->in_filter, oc->in_filter); | 
					
						
							|  |  |  |   int export_changed = ! filter_same(nc->out_filter, oc->out_filter); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* We treat a change in preferences by reimporting routes */ | 
					
						
							|  |  |  |   if (nc->preference != oc->preference) | 
					
						
							|  |  |  |     import_changed = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 22:57:51 +01:00
										 |  |  |   if (import_changed || export_changed) | 
					
						
							|  |  |  |     log(L_INFO "Reloading protocol %s", p->name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   /* If import filter changed, call reload hook */ | 
					
						
							|  |  |  |   if (import_changed && ! (p->reload_routes && p->reload_routes(p))) | 
					
						
							| 
									
										
										
										
											2010-02-06 19:19:09 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |       /* Now, the protocol is reconfigured. But route reload failed
 | 
					
						
							|  |  |  | 	 and we have to do regular protocol restart. */ | 
					
						
							| 
									
										
										
										
											2010-02-06 22:57:51 +01:00
										 |  |  |       log(L_INFO "Restarting protocol %s", p->name); | 
					
						
							| 
									
										
										
										
											2010-02-06 19:19:09 +01:00
										 |  |  |       p->disabled = 1; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |       p->down_code = PDC_CF_RESTART; | 
					
						
							| 
									
										
										
										
											2010-02-06 19:19:09 +01:00
										 |  |  |       proto_rethink_goal(p); | 
					
						
							|  |  |  |       p->disabled = 0; | 
					
						
							|  |  |  |       proto_rethink_goal(p); | 
					
						
							|  |  |  |       return 1; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (export_changed) | 
					
						
							|  |  |  |     proto_request_feeding(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return 1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * protos_commit - commit new protocol configuration | 
					
						
							|  |  |  |  * @new: new configuration | 
					
						
							|  |  |  |  * @old: old configuration or %NULL if it's boot time config | 
					
						
							|  |  |  |  * @force_reconfig: force restart of all protocols (used for example | 
					
						
							|  |  |  |  * when the router ID changes) | 
					
						
							| 
									
										
										
										
											2009-06-19 23:49:34 +02:00
										 |  |  |  * @type: type of reconfiguration (RECONFIG_SOFT or RECONFIG_HARD) | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Scan differences between @old and @new configuration and adjust all | 
					
						
							|  |  |  |  * protocol instances to conform to the new configuration. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * When a protocol exists in the new configuration, but it doesn't in the | 
					
						
							|  |  |  |  * original one, it's immediately started. When a collision with the other | 
					
						
							|  |  |  |  * running protocol would arise, the new protocol will be temporarily stopped | 
					
						
							|  |  |  |  * by the locking mechanism. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * When a protocol exists in the old configuration, but it doesn't in the | 
					
						
							|  |  |  |  * new one, it's shut down and deleted after the shutdown completes. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-06-19 23:49:34 +02:00
										 |  |  |  * When a protocol exists in both configurations, the core decides | 
					
						
							|  |  |  |  * whether it's possible to reconfigure it dynamically - it checks all | 
					
						
							|  |  |  |  * the core properties of the protocol (changes in filters are ignored | 
					
						
							|  |  |  |  * if type is RECONFIG_SOFT) and if they match, it asks the | 
					
						
							|  |  |  |  * reconfigure() hook of the protocol to see if the protocol is able | 
					
						
							|  |  |  |  * to switch to the new configuration.  If it isn't possible, the | 
					
						
							|  |  |  |  * protocol is shut down and a new instance is started with the new | 
					
						
							|  |  |  |  * configuration after the shutdown is completed. | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2009-06-19 23:49:34 +02:00
										 |  |  | protos_commit(struct config *new, struct config *old, int force_reconfig, int type) | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |   struct proto_config *oc, *nc; | 
					
						
							|  |  |  |   struct proto *p, *n; | 
					
						
							| 
									
										
										
										
											2011-11-07 00:31:23 +01:00
										 |  |  |   struct symbol *sym; | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |   DBG("protos_commit:\n"); | 
					
						
							|  |  |  |   if (old) | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |       WALK_LIST(oc, old->protos) | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2011-11-07 00:31:23 +01:00
										 |  |  | 	  p = oc->proto; | 
					
						
							| 
									
										
										
										
											2015-11-09 00:42:02 +01:00
										 |  |  | 	  sym = cf_find_symbol(new, oc->name); | 
					
						
							| 
									
										
										
										
											2000-01-16 17:40:26 +00:00
										 |  |  | 	  if (sym && sym->class == SYM_PROTO && !new->shutdown) | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	    { | 
					
						
							|  |  |  | 	      /* Found match, let's check if we can smoothly switch to new configuration */ | 
					
						
							| 
									
										
										
										
											2009-11-17 15:45:05 +01:00
										 |  |  | 	      /* No need to check description */ | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	      nc = sym->def; | 
					
						
							| 
									
										
										
										
											2010-02-06 19:19:09 +01:00
										 |  |  | 	      nc->proto = p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      /* We will try to reconfigure protocol p */ | 
					
						
							|  |  |  | 	      if (! force_reconfig && proto_reconfigure(p, oc, nc, type)) | 
					
						
							|  |  |  | 		continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	      /* Unsuccessful, we will restart it */ | 
					
						
							| 
									
										
										
										
											2010-02-06 22:57:51 +01:00
										 |  |  | 	      if (!p->disabled && !nc->disabled) | 
					
						
							|  |  |  | 		log(L_INFO "Restarting protocol %s", p->name); | 
					
						
							|  |  |  | 	      else if (p->disabled && !nc->disabled) | 
					
						
							|  |  |  | 		log(L_INFO "Enabling protocol %s", p->name); | 
					
						
							|  |  |  | 	      else if (!p->disabled && nc->disabled) | 
					
						
							|  |  |  | 		log(L_INFO "Disabling protocol %s", p->name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 	      p->down_code = nc->disabled ? PDC_CF_DISABLE : PDC_CF_RESTART; | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	      p->cf_new = nc; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2012-12-26 12:40:48 +01:00
										 |  |  | 	  else if (!new->shutdown) | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	    { | 
					
						
							| 
									
										
										
										
											2012-08-06 02:42:24 +02:00
										 |  |  | 	      log(L_INFO "Removing protocol %s", p->name); | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 	      p->down_code = PDC_CF_REMOVE; | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	      p->cf_new = NULL; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2012-08-06 02:42:24 +02:00
										 |  |  | 	  else /* global shutdown */ | 
					
						
							|  |  |  | 	    { | 
					
						
							|  |  |  | 	      p->down_code = PDC_CMD_SHUTDOWN; | 
					
						
							|  |  |  | 	      p->cf_new = NULL; | 
					
						
							|  |  |  | 	    } | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-06 02:42:24 +02:00
										 |  |  | 	  p->reconfiguring = 1; | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	  config_add_obstacle(old); | 
					
						
							|  |  |  | 	  proto_rethink_goal(p); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   WALK_LIST(nc, new->protos) | 
					
						
							|  |  |  |     if (!nc->proto) | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2012-12-26 12:40:48 +01:00
										 |  |  | 	if (old)		/* Not a first-time configuration */ | 
					
						
							| 
									
										
										
										
											2010-02-06 22:57:51 +01:00
										 |  |  | 	  log(L_INFO "Adding protocol %s", nc->name); | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	proto_init(nc); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |   DBG("\tdone\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   DBG("Protocol start\n"); | 
					
						
							| 
									
										
										
										
											2011-04-01 13:54:39 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* Start device protocol first */ | 
					
						
							|  |  |  |   if (initial_device_proto) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     proto_rethink_goal(initial_device_proto); | 
					
						
							|  |  |  |     initial_device_proto = NULL; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-12-27 12:56:23 +01:00
										 |  |  |   /* Determine router ID for the first time - it has to be here and not in
 | 
					
						
							|  |  |  |      global_commit() because it is postponed after start of device protocol */ | 
					
						
							|  |  |  |   if (!config->router_id) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       config->router_id = if_choose_router_id(config->router_id_from, 0); | 
					
						
							|  |  |  |       if (!config->router_id) | 
					
						
							|  |  |  | 	die("Cannot determine router ID, please configure it manually"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Start all other protocols */ | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |   WALK_LIST_DELSAFE(p, n, initial_proto_list) | 
					
						
							|  |  |  |     proto_rethink_goal(p); | 
					
						
							| 
									
										
										
										
											1998-06-03 08:38:53 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1998-10-17 11:05:18 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | proto_rethink_goal(struct proto *p) | 
					
						
							| 
									
										
										
										
											1998-10-17 11:05:18 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |   struct protocol *q; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   byte goal; | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (p->reconfiguring && p->core_state == FS_HUNGRY && p->proto_state == PS_DOWN) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       struct proto_config *nc = p->cf_new; | 
					
						
							|  |  |  |       DBG("%s has shut down for reconfiguration\n", p->name); | 
					
						
							| 
									
										
										
										
											2014-05-05 11:05:12 +02:00
										 |  |  |       p->cf->proto = NULL; | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |       config_del_obstacle(p->cf->global); | 
					
						
							|  |  |  |       rem_node(&p->n); | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  |       rem_node(&p->glob_node); | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |       mb_free(p); | 
					
						
							|  |  |  |       if (!nc) | 
					
						
							|  |  |  | 	return; | 
					
						
							| 
									
										
										
										
											2000-03-12 22:53:05 +00:00
										 |  |  |       p = proto_init(nc); | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Determine what state we want to reach */ | 
					
						
							| 
									
										
										
										
											2000-01-16 17:40:26 +00:00
										 |  |  |   if (p->disabled || p->reconfiguring) | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |     goal = PS_DOWN; | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |     goal = PS_UP; | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   q = p->proto; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   if (goal == PS_UP) 			/* Going up */ | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |       if (p->proto_state == PS_DOWN && p->core_state == FS_HUNGRY) | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  DBG("Kicking %s up\n", p->name); | 
					
						
							| 
									
										
										
										
											2000-03-07 21:50:21 +00:00
										 |  |  | 	  PD(p, "Starting"); | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  | 	  proto_init_instance(p); | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | 	  proto_notify_state(p, (q->start ? q->start(p) : PS_UP)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else 					/* Going down */ | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (p->proto_state == PS_START || p->proto_state == PS_UP) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  DBG("Kicking %s down\n", p->name); | 
					
						
							| 
									
										
										
										
											2000-03-07 21:50:21 +00:00
										 |  |  | 	  PD(p, "Shutting down"); | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | 	  proto_notify_state(p, (q->shutdown ? q->shutdown(p) : PS_DOWN)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * DOC: Graceful restart recovery | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Graceful restart of a router is a process when the routing plane (e.g. BIRD) | 
					
						
							|  |  |  |  * restarts but both the forwarding plane (e.g kernel routing table) and routing | 
					
						
							|  |  |  |  * neighbors keep proper routes, and therefore uninterrupted packet forwarding | 
					
						
							|  |  |  |  * is maintained. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * BIRD implements graceful restart recovery by deferring export of routes to | 
					
						
							|  |  |  |  * protocols until routing tables are refilled with the expected content. After | 
					
						
							|  |  |  |  * start, protocols generate routes as usual, but routes are not propagated to | 
					
						
							|  |  |  |  * them, until protocols report that they generated all routes. After that, | 
					
						
							|  |  |  |  * graceful restart recovery is finished and the export (and the initial feed) | 
					
						
							|  |  |  |  * to protocols is enabled. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * When graceful restart recovery need is detected during initialization, then | 
					
						
							|  |  |  |  * enabled protocols are marked with @gr_recovery flag before start. Such | 
					
						
							|  |  |  |  * protocols then decide how to proceed with graceful restart, participation is | 
					
						
							|  |  |  |  * voluntary. Protocols could lock the recovery by proto_graceful_restart_lock() | 
					
						
							|  |  |  |  * (stored in @gr_lock flag), which means that they want to postpone the end of | 
					
						
							|  |  |  |  * the recovery until they converge and then unlock it. They also could set | 
					
						
							|  |  |  |  * @gr_wait before advancing to %PS_UP, which means that the core should defer | 
					
						
							|  |  |  |  * route export to that protocol until the end of the recovery. This should be | 
					
						
							|  |  |  |  * done by protocols that expect their neigbors to keep the proper routes | 
					
						
							|  |  |  |  * (kernel table, BGP sessions with BGP graceful restart capability). | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The graceful restart recovery is finished when either all graceful restart | 
					
						
							|  |  |  |  * locks are unlocked or when graceful restart wait timer fires. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | static void graceful_restart_done(struct timer *t); | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * graceful_restart_recovery - request initial graceful restart recovery | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Called by the platform initialization code if the need for recovery | 
					
						
							|  |  |  |  * after graceful restart is detected during boot. Have to be called | 
					
						
							|  |  |  |  * before protos_commit(). | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | void | 
					
						
							|  |  |  | graceful_restart_recovery(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   graceful_restart_state = GRS_INIT; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * graceful_restart_init - initialize graceful restart | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * When graceful restart recovery was requested, the function starts an active | 
					
						
							|  |  |  |  * phase of the recovery and initializes graceful restart wait timer. The | 
					
						
							|  |  |  |  * function have to be called after protos_commit(). | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | void | 
					
						
							|  |  |  | graceful_restart_init(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!graceful_restart_state) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   log(L_INFO "Graceful restart started"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!graceful_restart_locks) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       graceful_restart_done(NULL); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   graceful_restart_state = GRS_ACTIVE; | 
					
						
							|  |  |  |   gr_wait_timer = tm_new(proto_pool); | 
					
						
							|  |  |  |   gr_wait_timer->hook = graceful_restart_done; | 
					
						
							|  |  |  |   tm_start(gr_wait_timer, config->gr_wait); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * graceful_restart_done - finalize graceful restart | 
					
						
							| 
									
										
										
										
											2016-05-12 15:49:44 +02:00
										 |  |  |  * @t: unused | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * When there are no locks on graceful restart, the functions finalizes the | 
					
						
							|  |  |  |  * graceful restart recovery. Protocols postponing route export until the end of | 
					
						
							|  |  |  |  * the recovery are awakened and the export to them is enabled. All other | 
					
						
							|  |  |  |  * related state is cleared. The function is also called when the graceful | 
					
						
							|  |  |  |  * restart wait timer fires (but there are still some locks). | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | static void | 
					
						
							|  |  |  | graceful_restart_done(struct timer *t UNUSED) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct proto *p; | 
					
						
							|  |  |  |   node *n; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   log(L_INFO "Graceful restart done"); | 
					
						
							|  |  |  |   graceful_restart_state = GRS_DONE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WALK_LIST2(p, n, proto_list, glob_node) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (!p->gr_recovery) | 
					
						
							|  |  |  | 	continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Resume postponed export of routes */ | 
					
						
							|  |  |  |       if ((p->proto_state == PS_UP) && p->gr_wait) | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  |       { | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 	proto_want_export_up(p); | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  | 	proto_log_state_change(p); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |       /* Cleanup */ | 
					
						
							|  |  |  |       p->gr_recovery = 0; | 
					
						
							|  |  |  |       p->gr_wait = 0; | 
					
						
							|  |  |  |       p->gr_lock = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   graceful_restart_locks = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | graceful_restart_show_status(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (graceful_restart_state != GRS_ACTIVE) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   cli_msg(-24, "Graceful restart recovery in progress"); | 
					
						
							|  |  |  |   cli_msg(-24, "  Waiting for %d protocols to recover", graceful_restart_locks); | 
					
						
							|  |  |  |   cli_msg(-24, "  Wait timer is %d/%d", tm_remains(gr_wait_timer), config->gr_wait); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_graceful_restart_lock - lock graceful restart by protocol | 
					
						
							|  |  |  |  * @p: protocol instance | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function allows a protocol to postpone the end of graceful restart | 
					
						
							|  |  |  |  * recovery until it converges. The lock is removed when the protocol calls | 
					
						
							|  |  |  |  * proto_graceful_restart_unlock() or when the protocol is stopped. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The function have to be called during the initial phase of graceful restart | 
					
						
							|  |  |  |  * recovery and only for protocols that are part of graceful restart (i.e. their | 
					
						
							|  |  |  |  * @gr_recovery is set), which means it should be called from protocol start | 
					
						
							|  |  |  |  * hooks. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | void | 
					
						
							|  |  |  | proto_graceful_restart_lock(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ASSERT(graceful_restart_state == GRS_INIT); | 
					
						
							|  |  |  |   ASSERT(p->gr_recovery); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (p->gr_lock) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p->gr_lock = 1; | 
					
						
							|  |  |  |   graceful_restart_locks++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_graceful_restart_unlock - unlock graceful restart by protocol | 
					
						
							|  |  |  |  * @p: protocol instance | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function unlocks a lock from proto_graceful_restart_lock(). It is also | 
					
						
							|  |  |  |  * automatically called when the lock holding protocol went down. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | void | 
					
						
							|  |  |  | proto_graceful_restart_unlock(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   if (!p->gr_lock) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p->gr_lock = 0; | 
					
						
							|  |  |  |   graceful_restart_locks--; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if ((graceful_restart_state == GRS_ACTIVE) && !graceful_restart_locks) | 
					
						
							|  |  |  |     tm_start(gr_wait_timer, 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * protos_dump_all - dump status of all protocols | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function dumps status of all existing protocol instances to the | 
					
						
							|  |  |  |  * debug output. It involves printing of general status information | 
					
						
							|  |  |  |  * such as protocol states, its position on the protocol lists | 
					
						
							|  |  |  |  * and also calling of a dump() hook of the protocol to print | 
					
						
							|  |  |  |  * the internals. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											1998-07-09 19:36:52 +00:00
										 |  |  | void | 
					
						
							|  |  |  | protos_dump_all(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct proto *p; | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   struct announce_hook *a; | 
					
						
							| 
									
										
										
										
											1998-07-09 19:36:52 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   debug("Protocols:\n"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  |   WALK_LIST(p, active_proto_list) | 
					
						
							| 
									
										
										
										
											1998-07-09 19:36:52 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2000-01-18 11:01:03 +00:00
										 |  |  |       debug("  protocol %s state %s/%s\n", p->name, | 
					
						
							| 
									
										
										
										
											1999-03-03 19:33:54 +00:00
										 |  |  | 	    p_states[p->proto_state], c_states[p->core_state]); | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |       for (a = p->ahooks; a; a = a->next) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  debug("\tTABLE %s\n", a->table->name); | 
					
						
							|  |  |  | 	  if (a->in_filter) | 
					
						
							|  |  |  | 	    debug("\tInput filter: %s\n", filter_name(a->in_filter)); | 
					
						
							|  |  |  | 	  if (a->out_filter != FILTER_REJECT) | 
					
						
							|  |  |  | 	    debug("\tOutput filter: %s\n", filter_name(a->out_filter)); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1998-11-29 22:01:33 +00:00
										 |  |  |       if (p->disabled) | 
					
						
							|  |  |  | 	debug("\tDISABLED\n"); | 
					
						
							| 
									
										
										
										
											1999-02-05 21:37:34 +00:00
										 |  |  |       else if (p->proto->dump) | 
					
						
							|  |  |  | 	p->proto->dump(p); | 
					
						
							| 
									
										
										
										
											1998-07-09 19:36:52 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											1998-10-17 11:05:18 +00:00
										 |  |  |   WALK_LIST(p, inactive_proto_list) | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |     debug("  inactive %s: state %s/%s\n", p->name, p_states[p->proto_state], c_states[p->core_state]); | 
					
						
							|  |  |  |   WALK_LIST(p, initial_proto_list) | 
					
						
							|  |  |  |     debug("  initial %s\n", p->name); | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  |   WALK_LIST(p, flush_proto_list) | 
					
						
							|  |  |  |     debug("  flushing %s\n", p->name); | 
					
						
							| 
									
										
										
										
											1998-07-09 19:36:52 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_build - make a single protocol available | 
					
						
							|  |  |  |  * @p: the protocol | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * After the platform specific initialization code uses protos_build() | 
					
						
							|  |  |  |  * to add all the standard protocols, it should call proto_build() for | 
					
						
							| 
									
										
										
										
											2000-06-07 12:29:08 +00:00
										 |  |  |  * all platform specific protocols to inform the core that they exist. | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2000-04-01 10:19:47 +00:00
										 |  |  | void | 
					
						
							|  |  |  | proto_build(struct protocol *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   add_tail(&protocol_list, &p->n); | 
					
						
							|  |  |  |   if (p->attr_class) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       ASSERT(!attr_class_to_protocol[p->attr_class]); | 
					
						
							|  |  |  |       attr_class_to_protocol[p->attr_class] = p; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-11-19 22:33:48 +01:00
										 |  |  | /* FIXME: convert this call to some protocol hook */ | 
					
						
							|  |  |  | extern void bfd_init_all(void); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * protos_build - build a protocol list | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This function is called during BIRD startup to insert | 
					
						
							|  |  |  |  * all standard protocols to the global protocol list. Insertion | 
					
						
							|  |  |  |  * of platform specific protocols (such as the kernel syncer) | 
					
						
							|  |  |  |  * is in the domain of competence of the platform dependent | 
					
						
							|  |  |  |  * startup code. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											1998-10-18 11:53:21 +00:00
										 |  |  | void | 
					
						
							|  |  |  | protos_build(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   init_list(&protocol_list); | 
					
						
							| 
									
										
										
										
											2000-01-17 12:38:07 +00:00
										 |  |  |   init_list(&proto_list); | 
					
						
							|  |  |  |   init_list(&active_proto_list); | 
					
						
							|  |  |  |   init_list(&inactive_proto_list); | 
					
						
							|  |  |  |   init_list(&initial_proto_list); | 
					
						
							|  |  |  |   init_list(&flush_proto_list); | 
					
						
							| 
									
										
										
										
											2000-04-01 10:19:47 +00:00
										 |  |  |   proto_build(&proto_device); | 
					
						
							| 
									
										
										
										
											2011-03-13 11:33:50 +01:00
										 |  |  | #ifdef CONFIG_RADV
 | 
					
						
							|  |  |  |   proto_build(&proto_radv); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											1999-01-10 00:26:11 +00:00
										 |  |  | #ifdef CONFIG_RIP
 | 
					
						
							| 
									
										
										
										
											2000-04-01 10:19:47 +00:00
										 |  |  |   proto_build(&proto_rip); | 
					
						
							| 
									
										
										
										
											1999-01-10 00:26:11 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #ifdef CONFIG_STATIC
 | 
					
						
							| 
									
										
										
										
											2000-04-01 10:19:47 +00:00
										 |  |  |   proto_build(&proto_static); | 
					
						
							| 
									
										
										
										
											1999-03-09 22:27:43 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #ifdef CONFIG_OSPF
 | 
					
						
							| 
									
										
										
										
											2000-04-01 10:19:47 +00:00
										 |  |  |   proto_build(&proto_ospf); | 
					
						
							| 
									
										
										
										
											2000-01-17 00:19:58 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #ifdef CONFIG_PIPE
 | 
					
						
							| 
									
										
										
										
											2000-04-01 10:19:47 +00:00
										 |  |  |   proto_build(&proto_pipe); | 
					
						
							| 
									
										
										
										
											2000-03-19 22:09:07 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #ifdef CONFIG_BGP
 | 
					
						
							| 
									
										
										
										
											2000-04-01 10:19:47 +00:00
										 |  |  |   proto_build(&proto_bgp); | 
					
						
							| 
									
										
										
										
											1999-01-10 00:26:11 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-11-19 22:33:48 +01:00
										 |  |  | #ifdef CONFIG_BFD
 | 
					
						
							| 
									
										
										
										
											2013-09-16 23:57:40 +02:00
										 |  |  |   proto_build(&proto_bfd); | 
					
						
							| 
									
										
										
										
											2013-11-19 22:33:48 +01:00
										 |  |  |   bfd_init_all(); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-04-28 18:01:40 +02:00
										 |  |  | #ifdef CONFIG_BABEL
 | 
					
						
							|  |  |  |   proto_build(&proto_babel); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-09-16 23:57:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |   proto_pool = rp_new(&root_pool, "Protocols"); | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  |   proto_flush_event = ev_new(proto_pool); | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  |   proto_flush_event->hook = proto_flush_loop; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |   proto_shutdown_timer = tm_new(proto_pool); | 
					
						
							|  |  |  |   proto_shutdown_timer->hook = proto_shutdown_loop; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-05-19 10:46:26 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | proto_feed_more(void *P) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct proto *p = P; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   if (p->export_state != ES_FEEDING) | 
					
						
							| 
									
										
										
										
											2000-05-19 18:03:53 +00:00
										 |  |  |     return; | 
					
						
							| 
									
										
										
										
											2008-11-22 01:12:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   DBG("Feeding protocol %s continued\n", p->name); | 
					
						
							| 
									
										
										
										
											2000-05-19 10:46:26 +00:00
										 |  |  |   if (rt_feed_baby(p)) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |       DBG("Feeding protocol %s finished\n", p->name); | 
					
						
							|  |  |  |       p->export_state = ES_READY; | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  |       proto_log_state_change(p); | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-29 18:27:13 +02:00
										 |  |  |       if (p->feed_end) | 
					
						
							|  |  |  | 	p->feed_end(p); | 
					
						
							| 
									
										
										
										
											2000-05-19 10:46:26 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       p->attn->hook = proto_feed_more; | 
					
						
							|  |  |  |       ev_schedule(p->attn);		/* Will continue later... */ | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-04-27 22:28:49 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  | proto_feed_initial(void *P) | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   struct proto *p = P; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   if (p->export_state != ES_FEEDING) | 
					
						
							| 
									
										
										
										
											2008-11-22 01:12:22 +01:00
										 |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |   DBG("Feeding protocol %s\n", p->name); | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |   if_feed_baby(p); | 
					
						
							| 
									
										
										
										
											2000-05-19 10:46:26 +00:00
										 |  |  |   proto_feed_more(P); | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 12:24:55 +01:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  | proto_schedule_feed(struct proto *p, int initial) | 
					
						
							| 
									
										
										
										
											2008-12-08 12:24:55 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   DBG("%s: Scheduling meal\n", p->name); | 
					
						
							| 
									
										
										
										
											2009-12-14 01:32:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   p->export_state = ES_FEEDING; | 
					
						
							|  |  |  |   p->refeeding = !initial; | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  |   p->attn->hook = initial ? proto_feed_initial : proto_feed_more; | 
					
						
							| 
									
										
										
										
											2008-12-08 12:24:55 +01:00
										 |  |  |   ev_schedule(p->attn); | 
					
						
							| 
									
										
										
										
											2015-03-29 18:27:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (p->feed_begin) | 
					
						
							|  |  |  |     p->feed_begin(p, initial); | 
					
						
							| 
									
										
										
										
											2008-12-08 12:24:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Flushing loop is responsible for flushing routes and protocols | 
					
						
							|  |  |  |  * after they went down. It runs in proto_flush_event. At the start of | 
					
						
							|  |  |  |  * one round, protocols waiting to flush are marked in | 
					
						
							|  |  |  |  * proto_schedule_flush_loop(). At the end of the round (when routing | 
					
						
							|  |  |  |  * table flush is complete), marked protocols are flushed and a next | 
					
						
							|  |  |  |  * round may start. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static int flush_loop_state;	/* 1 -> running */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_schedule_flush_loop(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct proto *p; | 
					
						
							| 
									
										
										
										
											2013-07-24 14:11:12 +02:00
										 |  |  |   struct announce_hook *h; | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (flush_loop_state) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  |   flush_loop_state = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WALK_LIST(p, flush_proto_list) | 
					
						
							| 
									
										
										
										
											2013-07-24 14:11:12 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  |     p->flushing = 1; | 
					
						
							| 
									
										
										
										
											2013-07-24 14:11:12 +02:00
										 |  |  |     for (h=p->ahooks; h; h=h->next) | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |       rt_mark_for_prune(h->table); | 
					
						
							| 
									
										
										
										
											2013-07-24 14:11:12 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   ev_schedule(proto_flush_event); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_flush_loop(void *unused UNUSED) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct proto *p; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (! rt_prune_loop()) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       /* Rtable pruning is not finished */ | 
					
						
							|  |  |  |       ev_schedule(proto_flush_event); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-14 16:25:22 +02:00
										 |  |  |   rt_prune_sources(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  |  again: | 
					
						
							|  |  |  |   WALK_LIST(p, flush_proto_list) | 
					
						
							|  |  |  |     if (p->flushing) | 
					
						
							|  |  |  |       { | 
					
						
							|  |  |  | 	/* This will flush interfaces in the same manner
 | 
					
						
							|  |  |  | 	   like rt_prune_all() flushes routes */ | 
					
						
							|  |  |  | 	if (p->proto == &proto_unix_iface) | 
					
						
							|  |  |  | 	  if_flush_ifaces(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	DBG("Flushing protocol %s\n", p->name); | 
					
						
							|  |  |  | 	p->flushing = 0; | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  | 	p->core_state = FS_HUNGRY; | 
					
						
							|  |  |  | 	proto_relink(p); | 
					
						
							|  |  |  | 	proto_log_state_change(p); | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  | 	if (p->proto_state == PS_DOWN) | 
					
						
							|  |  |  | 	  proto_fell_down(p); | 
					
						
							|  |  |  | 	goto again; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* This round finished, perhaps there will be another one */ | 
					
						
							|  |  |  |   flush_loop_state = 0; | 
					
						
							|  |  |  |   if (!EMPTY_LIST(flush_proto_list)) | 
					
						
							|  |  |  |     proto_schedule_flush_loop(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-24 23:39:57 +02:00
										 |  |  | /* Temporary hack to propagate restart to BGP */ | 
					
						
							|  |  |  | int proto_restart; | 
					
						
							| 
									
										
										
										
											2012-03-28 18:40:04 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | static void | 
					
						
							|  |  |  | proto_shutdown_loop(struct timer *t UNUSED) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct proto *p, *p_next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   WALK_LIST_DELSAFE(p, p_next, active_proto_list) | 
					
						
							|  |  |  |     if (p->down_sched) | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2012-04-24 23:39:57 +02:00
										 |  |  | 	proto_restart = (p->down_sched == PDS_RESTART); | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	p->disabled = 1; | 
					
						
							|  |  |  | 	proto_rethink_goal(p); | 
					
						
							| 
									
										
										
										
											2012-04-24 23:39:57 +02:00
										 |  |  | 	if (proto_restart) | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 	  { | 
					
						
							|  |  |  | 	    p->disabled = 0; | 
					
						
							|  |  |  | 	    proto_rethink_goal(p); | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void | 
					
						
							|  |  |  | proto_schedule_down(struct proto *p, byte restart, byte code) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   /* Does not work for other states (even PS_START) */ | 
					
						
							|  |  |  |   ASSERT(p->proto_state == PS_UP); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Scheduled restart may change to shutdown, but not otherwise */ | 
					
						
							|  |  |  |   if (p->down_sched == PDS_DISABLE) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p->down_sched = restart ? PDS_RESTART : PDS_DISABLE; | 
					
						
							|  |  |  |   p->down_code = code; | 
					
						
							|  |  |  |   tm_start_max(proto_shutdown_timer, restart ? 2 : 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_request_feeding - request feeding routes to the protocol | 
					
						
							|  |  |  |  * @p: given protocol  | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Sometimes it is needed to send again all routes to the | 
					
						
							|  |  |  |  * protocol. This is called feeding and can be requested by this | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |  * function. This would cause protocol export state transition | 
					
						
							|  |  |  |  * to ES_FEEDING (during feeding) and when completed, it will | 
					
						
							|  |  |  |  * switch back to ES_READY. This function can be called even | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  |  * when feeding is already running, in that case it is restarted. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | void | 
					
						
							|  |  |  | proto_request_feeding(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ASSERT(p->proto_state == PS_UP); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  |   /* Do nothing if we are still waiting for feeding */ | 
					
						
							|  |  |  |   if (p->export_state == ES_DOWN) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  |   /* If we are already feeding, we want to restart it */ | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   if (p->export_state == ES_FEEDING) | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |       /* Unless feeding is in initial state */ | 
					
						
							|  |  |  |       if (p->attn->hook == proto_feed_initial) | 
					
						
							|  |  |  | 	return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       rt_feed_baby_abort(p); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   /* FIXME: This should be changed for better support of multitable protos */ | 
					
						
							|  |  |  |   struct announce_hook *ah; | 
					
						
							|  |  |  |   for (ah = p->ahooks; ah; ah = ah->next) | 
					
						
							|  |  |  |     proto_reset_limit(ah->out_limit); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Hack: reset exp_routes during refeed, and do not decrease it later */ | 
					
						
							|  |  |  |   p->stats.exp_routes = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  |   proto_schedule_feed(p, 0); | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  |   proto_log_state_change(p); | 
					
						
							| 
									
										
										
										
											2009-11-26 20:47:59 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | static const char * | 
					
						
							|  |  |  | proto_limit_name(struct proto_limit *l) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   const char *actions[] = { | 
					
						
							|  |  |  |     [PLA_WARN] = "warn", | 
					
						
							|  |  |  |     [PLA_BLOCK] = "block", | 
					
						
							|  |  |  |     [PLA_RESTART] = "restart", | 
					
						
							|  |  |  |     [PLA_DISABLE] = "disable", | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return actions[l->action]; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * proto_notify_limit: notify about limit hit and take appropriate action | 
					
						
							|  |  |  |  * @ah: announce hook | 
					
						
							|  |  |  |  * @l: limit being hit | 
					
						
							| 
									
										
										
										
											2013-01-10 13:07:33 +01:00
										 |  |  |  * @dir: limit direction (PLD_*) | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |  * @rt_count: the number of routes  | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |  * | 
					
						
							|  |  |  |  * The function is called by the route processing core when limit @l | 
					
						
							|  |  |  |  * is breached. It activates the limit and tooks appropriate action | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |  * according to @l->action. | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2013-01-10 13:07:33 +01:00
										 |  |  | proto_notify_limit(struct announce_hook *ah, struct proto_limit *l, int dir, u32 rt_count) | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2013-01-10 13:07:33 +01:00
										 |  |  |   const char *dir_name[PLD_MAX] = { "receive", "import" , "export" }; | 
					
						
							|  |  |  |   const byte dir_down[PLD_MAX] = { PDC_RX_LIMIT_HIT, PDC_IN_LIMIT_HIT, PDC_OUT_LIMIT_HIT }; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |   struct proto *p = ah->proto; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |   if (l->state == PLS_BLOCKED) | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-24 23:39:57 +02:00
										 |  |  |   /* For warning action, we want the log message every time we hit the limit */ | 
					
						
							|  |  |  |   if (!l->state || ((l->action == PLA_WARN) && (rt_count == l->limit))) | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |     log(L_WARN "Protocol %s hits route %s limit (%d), action: %s", | 
					
						
							| 
									
										
										
										
											2013-01-10 13:07:33 +01:00
										 |  |  | 	p->name, dir_name[dir], l->limit, proto_limit_name(l)); | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |   switch (l->action) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case PLA_WARN: | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |       l->state = PLS_ACTIVE; | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case PLA_BLOCK: | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |       l->state = PLS_BLOCKED; | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case PLA_RESTART: | 
					
						
							|  |  |  |     case PLA_DISABLE: | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |       l->state = PLS_BLOCKED; | 
					
						
							| 
									
										
										
										
											2014-04-27 00:46:32 +02:00
										 |  |  |       if (p->proto_state == PS_UP) | 
					
						
							|  |  |  | 	proto_schedule_down(p, l->action == PLA_RESTART, dir_down[dir]); | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-04-27 00:46:32 +02:00
										 |  |  | void | 
					
						
							|  |  |  | proto_verify_limits(struct announce_hook *ah) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct proto_limit *l; | 
					
						
							|  |  |  |   struct proto_stats *stats = ah->stats; | 
					
						
							|  |  |  |   u32 all_routes = stats->imp_routes + stats->filt_routes; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   l = ah->rx_limit; | 
					
						
							|  |  |  |   if (l && (all_routes > l->limit)) | 
					
						
							|  |  |  |     proto_notify_limit(ah, l, PLD_RX, all_routes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   l = ah->in_limit; | 
					
						
							|  |  |  |   if (l && (stats->imp_routes > l->limit)) | 
					
						
							|  |  |  |     proto_notify_limit(ah, l, PLD_IN, stats->imp_routes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   l = ah->out_limit; | 
					
						
							|  |  |  |   if (l && (stats->exp_routes > l->limit)) | 
					
						
							|  |  |  |     proto_notify_limit(ah, l, PLD_OUT, stats->exp_routes); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_want_core_up(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ASSERT(p->core_state == FS_HUNGRY); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!p->proto->multitable) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       p->main_source = rt_get_source(p, 0); | 
					
						
							|  |  |  |       rt_lock_source(p->main_source); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /* Connect protocol to routing table */ | 
					
						
							|  |  |  |       p->main_ahook = proto_add_announce_hook(p, p->table, &p->stats); | 
					
						
							|  |  |  |       p->main_ahook->in_filter = p->cf->in_filter; | 
					
						
							|  |  |  |       p->main_ahook->out_filter = p->cf->out_filter; | 
					
						
							|  |  |  |       p->main_ahook->rx_limit = p->cf->rx_limit; | 
					
						
							|  |  |  |       p->main_ahook->in_limit = p->cf->in_limit; | 
					
						
							|  |  |  |       p->main_ahook->out_limit = p->cf->out_limit; | 
					
						
							|  |  |  |       p->main_ahook->in_keep_filtered = p->cf->in_keep_filtered; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       proto_reset_limit(p->main_ahook->rx_limit); | 
					
						
							|  |  |  |       proto_reset_limit(p->main_ahook->in_limit); | 
					
						
							|  |  |  |       proto_reset_limit(p->main_ahook->out_limit); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  |   p->core_state = FS_HAPPY; | 
					
						
							|  |  |  |   proto_relink(p); | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_want_export_up(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-04-07 11:48:25 +02:00
										 |  |  |   ASSERT(p->core_state == FS_HAPPY); | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   ASSERT(p->export_state == ES_DOWN); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   proto_link_ahooks(p); | 
					
						
							|  |  |  |   proto_schedule_feed(p, 1); /* Sets ES_FEEDING */ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_want_export_down(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   ASSERT(p->export_state != ES_DOWN); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Need to abort feeding */ | 
					
						
							|  |  |  |   if (p->export_state == ES_FEEDING) | 
					
						
							|  |  |  |     rt_feed_baby_abort(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   p->export_state = ES_DOWN; | 
					
						
							| 
									
										
										
										
											2016-04-07 01:10:24 +02:00
										 |  |  |   p->stats.exp_routes = 0; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   proto_unlink_ahooks(p); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_want_core_down(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2014-04-07 11:48:25 +02:00
										 |  |  |   ASSERT(p->core_state == FS_HAPPY); | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   ASSERT(p->export_state == ES_DOWN); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  |   p->core_state = FS_FLUSHING; | 
					
						
							|  |  |  |   proto_relink(p); | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   proto_schedule_flush_loop(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!p->proto->multitable) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       rt_unlock_source(p->main_source); | 
					
						
							|  |  |  |       p->main_source = NULL; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | proto_falling_down(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   p->gr_recovery = 0; | 
					
						
							|  |  |  |   p->gr_wait = 0; | 
					
						
							|  |  |  |   if (p->gr_lock) | 
					
						
							|  |  |  |     proto_graceful_restart_unlock(p); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-23 01:35:33 +01:00
										 |  |  | static void | 
					
						
							|  |  |  | proto_fell_down(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   DBG("Protocol %s down\n", p->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   u32 all_routes = p->stats.imp_routes + p->stats.filt_routes; | 
					
						
							|  |  |  |   if (all_routes != 0) | 
					
						
							|  |  |  |     log(L_ERR "Protocol %s is down but still has %d routes", p->name, all_routes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bzero(&p->stats, sizeof(struct proto_stats)); | 
					
						
							|  |  |  |   proto_free_ahooks(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (! p->proto->multitable) | 
					
						
							|  |  |  |     rt_unlock_table(p->table); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (p->proto->cleanup) | 
					
						
							|  |  |  |     p->proto->cleanup(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   proto_rethink_goal(p); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * proto_notify_state - notify core about protocol state change | 
					
						
							|  |  |  |  * @p: protocol the state of which has changed | 
					
						
							|  |  |  |  * @ps: the new status | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Whenever a state of a protocol changes due to some event internal | 
					
						
							|  |  |  |  * to the protocol (i.e., not inside a start() or shutdown() hook), | 
					
						
							|  |  |  |  * it should immediately notify the core about the change by calling | 
					
						
							|  |  |  |  * proto_notify_state() which will write the new state to the &proto | 
					
						
							|  |  |  |  * structure and take all the actions necessary to adapt to the new | 
					
						
							| 
									
										
										
										
											2008-12-08 12:24:55 +01:00
										 |  |  |  * state. State change to PS_DOWN immediately frees resources of protocol | 
					
						
							|  |  |  |  * and might execute start callback of protocol; therefore, | 
					
						
							|  |  |  |  * it should be used at tail positions of protocol callbacks. | 
					
						
							| 
									
										
										
										
											2000-06-02 13:42:36 +00:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | void | 
					
						
							|  |  |  | proto_notify_state(struct proto *p, unsigned ps) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   unsigned ops = p->proto_state; | 
					
						
							|  |  |  |   unsigned cs = p->core_state; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   unsigned es = p->export_state; | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   DBG("%s reporting state transition %s/%s -> */%s\n", p->name, c_states[cs], p_states[ops], p_states[ps]); | 
					
						
							|  |  |  |   if (ops == ps) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 12:24:55 +01:00
										 |  |  |   p->proto_state = ps; | 
					
						
							| 
									
										
										
										
											2013-11-24 22:22:24 +01:00
										 |  |  |   p->last_state_change = now; | 
					
						
							| 
									
										
										
										
											2008-12-08 12:24:55 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |   switch (ps) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |     case PS_START: | 
					
						
							|  |  |  |       ASSERT(ops == PS_DOWN || ops == PS_UP); | 
					
						
							|  |  |  |       ASSERT(cs == FS_HUNGRY || cs == FS_HAPPY); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (es != ES_DOWN) | 
					
						
							|  |  |  | 	proto_want_export_down(p); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case PS_UP: | 
					
						
							|  |  |  |       ASSERT(ops == PS_DOWN || ops == PS_START); | 
					
						
							|  |  |  |       ASSERT(cs == FS_HUNGRY || cs == FS_HAPPY); | 
					
						
							|  |  |  |       ASSERT(es == ES_DOWN); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (cs == FS_HUNGRY) | 
					
						
							|  |  |  | 	proto_want_core_up(p); | 
					
						
							|  |  |  |       if (!p->gr_wait) | 
					
						
							|  |  |  | 	proto_want_export_up(p); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case PS_STOP: | 
					
						
							|  |  |  |       ASSERT(ops == PS_START || ops == PS_UP); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       p->down_sched = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (es != ES_DOWN) | 
					
						
							|  |  |  | 	proto_want_export_down(p); | 
					
						
							|  |  |  |       if (cs == FS_HAPPY) | 
					
						
							|  |  |  | 	proto_want_core_down(p); | 
					
						
							|  |  |  |       proto_falling_down(p); | 
					
						
							|  |  |  |       break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |     case PS_DOWN: | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |       p->down_code = 0; | 
					
						
							|  |  |  |       p->down_sched = 0; | 
					
						
							| 
									
										
										
										
											2009-02-12 13:43:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |       if (es != ES_DOWN) | 
					
						
							|  |  |  | 	proto_want_export_down(p); | 
					
						
							|  |  |  |       if (cs == FS_HAPPY) | 
					
						
							|  |  |  | 	proto_want_core_down(p); | 
					
						
							|  |  |  |       if (ops != PS_STOP) | 
					
						
							|  |  |  | 	proto_falling_down(p); | 
					
						
							| 
									
										
										
										
											2012-08-14 16:25:22 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-08 12:24:55 +01:00
										 |  |  |       neigh_prune(); // FIXME convert neighbors to resource?
 | 
					
						
							|  |  |  |       rfree(p->pool); | 
					
						
							|  |  |  |       p->pool = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |       if (cs == FS_HUNGRY)		/* Shutdown finished */ | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  | 	  proto_log_state_change(p); | 
					
						
							| 
									
										
										
										
											2000-01-16 16:44:50 +00:00
										 |  |  | 	  proto_fell_down(p); | 
					
						
							|  |  |  | 	  return;			/* The protocol might have ceased to exist */ | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |       break; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |     default: | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |       bug("%s: Invalid state %d", p->name, ps); | 
					
						
							| 
									
										
										
										
											1999-02-11 22:59:06 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   proto_log_state_change(p); | 
					
						
							| 
									
										
										
										
											1998-10-18 11:53:21 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											1999-02-13 19:15:28 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-11-30 12:57:14 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  *  CLI Commands | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static char * | 
					
						
							|  |  |  | proto_state_name(struct proto *p) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #define P(x,y) ((x << 4) | y)
 | 
					
						
							|  |  |  |   switch (P(p->proto_state, p->core_state)) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     case P(PS_DOWN, FS_HUNGRY):		return "down"; | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |     case P(PS_START, FS_HUNGRY): | 
					
						
							|  |  |  |     case P(PS_START, FS_HAPPY):		return "start"; | 
					
						
							|  |  |  |     case P(PS_UP, FS_HAPPY): | 
					
						
							|  |  |  |       switch (p->export_state) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	case ES_DOWN:			return "wait"; | 
					
						
							|  |  |  | 	case ES_FEEDING:		return "feed"; | 
					
						
							|  |  |  | 	case ES_READY:			return "up"; | 
					
						
							|  |  |  | 	default:      			return "???"; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2014-03-24 12:32:12 +01:00
										 |  |  |     case P(PS_STOP, FS_HUNGRY): | 
					
						
							|  |  |  |     case P(PS_STOP, FS_FLUSHING):	return "stop"; | 
					
						
							| 
									
										
										
										
											1999-11-30 12:57:14 +00:00
										 |  |  |     case P(PS_DOWN, FS_FLUSHING):	return "flush"; | 
					
						
							|  |  |  |     default:      			return "???"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | #undef P
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-13 10:44:46 +01:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2012-11-15 01:29:01 +01:00
										 |  |  | proto_show_stats(struct proto_stats *s, int in_keep_filtered) | 
					
						
							| 
									
										
										
										
											2010-02-13 10:44:46 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-11-15 01:29:01 +01:00
										 |  |  |   if (in_keep_filtered) | 
					
						
							|  |  |  |     cli_msg(-1006, "  Routes:         %u imported, %u filtered, %u exported, %u preferred",  | 
					
						
							|  |  |  | 	    s->imp_routes, s->filt_routes, s->exp_routes, s->pref_routes); | 
					
						
							| 
									
										
										
										
											2012-11-10 14:26:13 +01:00
										 |  |  |   else | 
					
						
							|  |  |  |     cli_msg(-1006, "  Routes:         %u imported, %u exported, %u preferred",  | 
					
						
							|  |  |  | 	    s->imp_routes, s->exp_routes, s->pref_routes); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-13 10:44:46 +01:00
										 |  |  |   cli_msg(-1006, "  Route change stats:     received   rejected   filtered    ignored   accepted"); | 
					
						
							|  |  |  |   cli_msg(-1006, "    Import updates:     %10u %10u %10u %10u %10u", | 
					
						
							|  |  |  | 	  s->imp_updates_received, s->imp_updates_invalid, | 
					
						
							|  |  |  | 	  s->imp_updates_filtered, s->imp_updates_ignored, | 
					
						
							|  |  |  | 	  s->imp_updates_accepted); | 
					
						
							|  |  |  |   cli_msg(-1006, "    Import withdraws:   %10u %10u        --- %10u %10u", | 
					
						
							|  |  |  | 	  s->imp_withdraws_received, s->imp_withdraws_invalid, | 
					
						
							|  |  |  | 	  s->imp_withdraws_ignored, s->imp_withdraws_accepted); | 
					
						
							|  |  |  |   cli_msg(-1006, "    Export updates:     %10u %10u %10u        --- %10u", | 
					
						
							|  |  |  | 	  s->exp_updates_received, s->exp_updates_rejected, | 
					
						
							|  |  |  | 	  s->exp_updates_filtered, s->exp_updates_accepted); | 
					
						
							|  |  |  |   cli_msg(-1006, "    Export withdraws:   %10u        ---        ---        --- %10u", | 
					
						
							|  |  |  | 	  s->exp_withdraws_received, s->exp_withdraws_accepted); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | void | 
					
						
							|  |  |  | proto_show_limit(struct proto_limit *l, const char *dsc) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |   if (!l) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   cli_msg(-1006, "  %-16s%d%s", dsc, l->limit, l->state ? " [HIT]" : ""); | 
					
						
							|  |  |  |   cli_msg(-1006, "    Action:       %s", proto_limit_name(l)); | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  | void | 
					
						
							|  |  |  | proto_show_basic_info(struct proto *p) | 
					
						
							| 
									
										
										
										
											2010-02-13 10:44:46 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   // cli_msg(-1006, "  Table:          %s", p->table->name);
 | 
					
						
							|  |  |  |   cli_msg(-1006, "  Preference:     %d", p->preference); | 
					
						
							|  |  |  |   cli_msg(-1006, "  Input filter:   %s", filter_name(p->cf->in_filter)); | 
					
						
							|  |  |  |   cli_msg(-1006, "  Output filter:  %s", filter_name(p->cf->out_filter)); | 
					
						
							| 
									
										
										
										
											2010-02-13 10:44:46 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-03-20 14:07:12 +01:00
										 |  |  |   if (graceful_restart_state == GRS_ACTIVE) | 
					
						
							|  |  |  |     cli_msg(-1006, "  GR recovery:   %s%s", | 
					
						
							|  |  |  | 	    p->gr_lock ? " pending" : "", | 
					
						
							|  |  |  | 	    p->gr_wait ? " waiting" : ""); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-10 13:07:33 +01:00
										 |  |  |   proto_show_limit(p->cf->rx_limit, "Receive limit:"); | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |   proto_show_limit(p->cf->in_limit, "Import limit:"); | 
					
						
							| 
									
										
										
										
											2012-04-24 23:39:57 +02:00
										 |  |  |   proto_show_limit(p->cf->out_limit, "Export limit:"); | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |   if (p->proto_state != PS_DOWN) | 
					
						
							| 
									
										
										
										
											2012-11-15 01:29:01 +01:00
										 |  |  |     proto_show_stats(&p->stats, p->cf->in_keep_filtered); | 
					
						
							| 
									
										
										
										
											2010-02-13 10:44:46 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_cmd_show(struct proto *p, uint verbose, int cnt) | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-03 00:19:24 +01:00
										 |  |  |   byte buf[256], tbuf[TM_DATETIME_BUFFER_SIZE]; | 
					
						
							| 
									
										
										
										
											1999-12-01 12:00:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  |   /* First protocol - show header */ | 
					
						
							|  |  |  |   if (!cnt) | 
					
						
							|  |  |  |     cli_msg(-2002, "name     proto    table    state  since       info"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-01 12:00:15 +00:00
										 |  |  |   buf[0] = 0; | 
					
						
							|  |  |  |   if (p->proto->get_status) | 
					
						
							|  |  |  |     p->proto->get_status(p, buf); | 
					
						
							| 
									
										
										
										
											2010-02-03 00:19:24 +01:00
										 |  |  |   tm_format_datetime(tbuf, &config->tf_proto, p->last_state_change); | 
					
						
							|  |  |  |   cli_msg(-1002, "%-8s %-8s %-8s %-5s  %-10s  %s", | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  | 	  p->name, | 
					
						
							|  |  |  | 	  p->proto->name, | 
					
						
							|  |  |  | 	  p->table->name, | 
					
						
							|  |  |  | 	  proto_state_name(p), | 
					
						
							| 
									
										
										
										
											2010-02-03 00:19:24 +01:00
										 |  |  | 	  tbuf, | 
					
						
							| 
									
										
										
										
											1999-12-01 12:00:15 +00:00
										 |  |  | 	  buf); | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |   if (verbose) | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2009-11-17 15:45:05 +01:00
										 |  |  |       if (p->cf->dsc) | 
					
						
							|  |  |  | 	cli_msg(-1006, "  Description:    %s", p->cf->dsc); | 
					
						
							| 
									
										
										
										
											2010-04-07 00:19:23 +02:00
										 |  |  |       if (p->cf->router_id) | 
					
						
							|  |  |  | 	cli_msg(-1006, "  Router ID:      %R", p->cf->router_id); | 
					
						
							| 
									
										
										
										
											2009-06-04 01:22:56 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-07 00:19:23 +02:00
										 |  |  |       if (p->proto->show_proto_info) | 
					
						
							|  |  |  | 	p->proto->show_proto_info(p); | 
					
						
							| 
									
										
										
										
											2012-03-15 11:58:08 +01:00
										 |  |  |       else | 
					
						
							|  |  |  | 	proto_show_basic_info(p); | 
					
						
							| 
									
										
										
										
											2010-04-07 00:19:23 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-04 01:22:56 +02:00
										 |  |  |       cli_msg(-1006, ""); | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-11-25 15:35:30 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_cmd_disable(struct proto *p, uint arg UNUSED, int cnt UNUSED) | 
					
						
							| 
									
										
										
										
											1999-11-25 15:35:30 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  |   if (p->disabled) | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  |       cli_msg(-8, "%s: already disabled", p->name); | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   log(L_INFO "Disabling protocol %s", p->name); | 
					
						
							|  |  |  |   p->disabled = 1; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |   p->down_code = PDC_CMD_DISABLE; | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  |   proto_rethink_goal(p); | 
					
						
							|  |  |  |   cli_msg(-9, "%s: disabled", p->name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_cmd_enable(struct proto *p, uint arg UNUSED, int cnt UNUSED) | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   if (!p->disabled) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       cli_msg(-10, "%s: already enabled", p->name); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   log(L_INFO "Enabling protocol %s", p->name); | 
					
						
							|  |  |  |   p->disabled = 0; | 
					
						
							|  |  |  |   proto_rethink_goal(p); | 
					
						
							|  |  |  |   cli_msg(-11, "%s: enabled", p->name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_cmd_restart(struct proto *p, uint arg UNUSED, int cnt UNUSED) | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   if (p->disabled) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       cli_msg(-8, "%s: already disabled", p->name); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   log(L_INFO "Restarting protocol %s", p->name); | 
					
						
							|  |  |  |   p->disabled = 1; | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |   p->down_code = PDC_CMD_RESTART; | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  |   proto_rethink_goal(p); | 
					
						
							|  |  |  |   p->disabled = 0; | 
					
						
							|  |  |  |   proto_rethink_goal(p); | 
					
						
							|  |  |  |   cli_msg(-12, "%s: restarted", p->name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_cmd_reload(struct proto *p, uint dir, int cnt UNUSED) | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   if (p->disabled) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       cli_msg(-8, "%s: already disabled", p->name); | 
					
						
							|  |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* If the protocol in not UP, it has no routes */ | 
					
						
							|  |  |  |   if (p->proto_state != PS_UP) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   log(L_INFO "Reloading protocol %s", p->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* re-importing routes */ | 
					
						
							|  |  |  |   if (dir != CMD_RELOAD_OUT) | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |     { | 
					
						
							|  |  |  |       if (! (p->reload_routes && p->reload_routes(p))) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 	  cli_msg(-8006, "%s: reload failed", p->name); | 
					
						
							|  |  |  | 	  return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       /*
 | 
					
						
							|  |  |  |        * Should be done before reload_routes() hook? | 
					
						
							|  |  |  |        * Perhaps, but these hooks work asynchronously. | 
					
						
							|  |  |  |        */ | 
					
						
							| 
									
										
										
										
											2012-04-21 21:05:36 +02:00
										 |  |  |       if (!p->proto->multitable) | 
					
						
							| 
									
										
										
										
											2013-01-10 13:07:33 +01:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 	  proto_reset_limit(p->main_ahook->rx_limit); | 
					
						
							|  |  |  | 	  proto_reset_limit(p->main_ahook->in_limit); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-04-15 15:28:29 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  |   /* re-exporting routes */ | 
					
						
							|  |  |  |   if (dir != CMD_RELOAD_IN) | 
					
						
							|  |  |  |     proto_request_feeding(p); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   cli_msg(-15, "%s: reloading", p->name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_cmd_debug(struct proto *p, uint mask, int cnt UNUSED) | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   p->debug = mask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_cmd_mrtdump(struct proto *p, uint mask, int cnt UNUSED) | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   p->mrtdump = mask; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_apply_cmd_symbol(struct symbol *s, void (* cmd)(struct proto *, uint, int), uint arg) | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   if (s->class != SYM_PROTO) | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |     { | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  |       cli_msg(9002, "%s is not a protocol", s->name); | 
					
						
							|  |  |  |       return; | 
					
						
							| 
									
										
										
										
											1999-11-30 14:04:09 +00:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   cmd(((struct proto_config *)s->def)->proto, arg, 0); | 
					
						
							| 
									
										
										
										
											1999-11-25 15:35:30 +00:00
										 |  |  |   cli_msg(0, ""); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											1999-12-03 11:40:45 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_apply_cmd_patt(char *patt, void (* cmd)(struct proto *, uint, int), uint arg) | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | { | 
					
						
							|  |  |  |   int cnt = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   node *nn; | 
					
						
							|  |  |  |   WALK_LIST(nn, proto_list) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       struct proto *p = SKIP_BACK(struct proto, glob_node, nn); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!patt || patmatch(patt, p->name)) | 
					
						
							|  |  |  | 	cmd(p, arg, cnt++); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!cnt) | 
					
						
							|  |  |  |     cli_msg(8003, "No protocols match"); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     cli_msg(0, ""); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void | 
					
						
							| 
									
										
										
										
											2015-05-19 08:53:34 +02:00
										 |  |  | proto_apply_cmd(struct proto_spec ps, void (* cmd)(struct proto *, uint, int), | 
					
						
							|  |  |  | 		int restricted, uint arg) | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-21 09:57:26 +01:00
										 |  |  |   if (restricted && cli_access_restricted()) | 
					
						
							|  |  |  |     return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-20 00:03:31 +01:00
										 |  |  |   if (ps.patt) | 
					
						
							|  |  |  |     proto_apply_cmd_patt(ps.ptr, cmd, arg); | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     proto_apply_cmd_symbol(ps.ptr, cmd, arg); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											1999-12-03 11:40:45 +00:00
										 |  |  | struct proto * | 
					
						
							|  |  |  | proto_get_named(struct symbol *sym, struct protocol *pr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   struct proto *p, *q; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (sym) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       if (sym->class != SYM_PROTO) | 
					
						
							|  |  |  | 	cf_error("%s: Not a protocol", sym->name); | 
					
						
							|  |  |  |       p = ((struct proto_config *)sym->def)->proto; | 
					
						
							|  |  |  |       if (!p || p->proto != pr) | 
					
						
							|  |  |  | 	cf_error("%s: Not a %s protocol", sym->name, pr->name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   else | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       p = NULL; | 
					
						
							| 
									
										
										
										
											2000-01-16 23:30:06 +00:00
										 |  |  |       WALK_LIST(q, active_proto_list) | 
					
						
							| 
									
										
										
										
											1999-12-03 11:40:45 +00:00
										 |  |  | 	if (q->proto == pr) | 
					
						
							|  |  |  | 	  { | 
					
						
							|  |  |  | 	    if (p) | 
					
						
							|  |  |  | 	      cf_error("There are multiple %s protocols running", pr->name); | 
					
						
							|  |  |  | 	    p = q; | 
					
						
							|  |  |  | 	  } | 
					
						
							|  |  |  |       if (!p) | 
					
						
							|  |  |  | 	cf_error("There is no %s protocol running", pr->name); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   return p; | 
					
						
							|  |  |  | } |