mirror of
				https://git.proxmox.com/git/mirror_frr
				synced 2025-10-31 21:46:33 +00:00 
			
		
		
		
	 7ee30f288e
			
		
	
	
		7ee30f288e
		
	
	
	
	
		
			
			Also modify `struct route_entry` to use nexthop_groups. Move ALL_NEXTHOPS loop to nexthop_group.h Signed-off-by: Donald Sharp <sharpd@cumulusnetworks.com>
		
			
				
	
	
		
			312 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			312 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Recursive Nexthop Iterator test.
 | |
|  * This tests the ALL_NEXTHOPS macro.
 | |
|  *
 | |
|  * Copyright (C) 2012 by Open Source Routing.
 | |
|  * Copyright (C) 2012 by Internet Systems Consortium, Inc. ("ISC")
 | |
|  *
 | |
|  * This file is part of Quagga
 | |
|  *
 | |
|  * Quagga is free software; you can redistribute it and/or modify it
 | |
|  * under the terms of the GNU General Public License as published by the
 | |
|  * Free Software Foundation; either version 2, or (at your option) any
 | |
|  * later version.
 | |
|  *
 | |
|  * Quagga is distributed in the hope that it will be useful, but
 | |
|  * WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 | |
|  * General Public License for more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License along
 | |
|  * with this program; see the file COPYING; if not, write to the Free Software
 | |
|  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
 | |
|  */
 | |
| 
 | |
| #include <zebra.h>
 | |
| #include "zebra/rib.h"
 | |
| #include "prng.h"
 | |
| 
 | |
| struct thread_master *master;
 | |
| static int verbose;
 | |
| 
 | |
| static void str_append(char **buf, const char *repr)
 | |
| {
 | |
| 	if (*buf) {
 | |
| 		*buf = realloc(*buf, strlen(*buf) + strlen(repr) + 1);
 | |
| 		assert(*buf);
 | |
| 		strncpy((*buf) + strlen(*buf), repr, strlen(repr) + 1);
 | |
| 	} else {
 | |
| 		*buf = strdup(repr);
 | |
| 		assert(*buf);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| static void str_appendf(char **buf, const char *format, ...)
 | |
| {
 | |
| 	va_list ap;
 | |
| 	int rv;
 | |
| 	char *pbuf;
 | |
| 
 | |
| 	va_start(ap, format);
 | |
| 	rv = vasprintf(&pbuf, format, ap);
 | |
| 	va_end(ap);
 | |
| 	assert(rv >= 0);
 | |
| 
 | |
| 	str_append(buf, pbuf);
 | |
| 	free(pbuf);
 | |
| }
 | |
| 
 | |
| /* This structure contains a nexthop chain
 | |
|  * and its expected representation */
 | |
| struct nexthop_chain {
 | |
| 	/* Head of the chain */
 | |
| 	struct nexthop_group head;
 | |
| 	/* Last nexthop in top chain */
 | |
| 	struct nexthop *current_top;
 | |
| 	/* Last nexthop in current recursive chain */
 | |
| 	struct nexthop *current_recursive;
 | |
| 	/* Expected string representation. */
 | |
| 	char *repr;
 | |
| };
 | |
| 
 | |
| static struct nexthop_chain *nexthop_chain_new(void)
 | |
| {
 | |
| 	struct nexthop_chain *rv;
 | |
| 
 | |
| 	rv = calloc(sizeof(*rv), 1);
 | |
| 	assert(rv);
 | |
| 	return rv;
 | |
| }
 | |
| 
 | |
| static void nexthop_chain_add_top(struct nexthop_chain *nc)
 | |
| {
 | |
| 	struct nexthop *nh;
 | |
| 
 | |
| 	nh = calloc(sizeof(*nh), 1);
 | |
| 	assert(nh);
 | |
| 
 | |
| 	if (nc->head.nexthop) {
 | |
| 		nc->current_top->next = nh;
 | |
| 		nh->prev = nc->current_top;
 | |
| 		nc->current_top = nh;
 | |
| 	} else {
 | |
| 		nc->head.nexthop = nc->current_top = nh;
 | |
| 	}
 | |
| 	nc->current_recursive = NULL;
 | |
| 	str_appendf(&nc->repr, "%p\n", nh);
 | |
| }
 | |
| 
 | |
| static void add_string_representation(char **repr, struct nexthop *nh)
 | |
| {
 | |
| 	struct nexthop *parent;
 | |
| 
 | |
| 	/* add indentations first */
 | |
| 	parent = nh->rparent;
 | |
| 	while (parent) {
 | |
| 		str_appendf(repr, "  ");
 | |
| 		parent = parent->rparent;
 | |
| 	}
 | |
| 	str_appendf(repr, "%p\n", nh);
 | |
| }
 | |
| 
 | |
| static void start_recursive_chain(struct nexthop_chain *nc, struct nexthop *nh)
 | |
| {
 | |
| 	SET_FLAG(nc->current_top->flags, NEXTHOP_FLAG_RECURSIVE);
 | |
| 	nc->current_top->resolved = nh;
 | |
| 	nh->rparent = nc->current_top;
 | |
| 	nc->current_recursive = nh;
 | |
| }
 | |
| static void nexthop_chain_add_recursive(struct nexthop_chain *nc)
 | |
| {
 | |
| 	struct nexthop *nh;
 | |
| 
 | |
| 	nh = calloc(sizeof(*nh), 1);
 | |
| 	assert(nh);
 | |
| 
 | |
| 	assert(nc->current_top);
 | |
| 	if (nc->current_recursive) {
 | |
| 		nc->current_recursive->next = nh;
 | |
| 		nh->prev = nc->current_recursive;
 | |
| 		nh->rparent = nc->current_recursive->rparent;
 | |
| 		nc->current_recursive = nh;
 | |
| 	} else
 | |
| 		start_recursive_chain(nc, nh);
 | |
| 
 | |
| 	add_string_representation(&nc->repr, nh);
 | |
| }
 | |
| 
 | |
| static void nexthop_chain_add_recursive_level(struct nexthop_chain *nc)
 | |
| {
 | |
| 	struct nexthop *nh;
 | |
| 
 | |
| 	nh = calloc(sizeof(*nh), 1);
 | |
| 	assert(nh);
 | |
| 
 | |
| 	assert(nc->current_top);
 | |
| 	if (nc->current_recursive) {
 | |
| 		SET_FLAG(nc->current_recursive->flags, NEXTHOP_FLAG_RECURSIVE);
 | |
| 		nc->current_recursive->resolved = nh;
 | |
| 		nh->rparent = nc->current_recursive;
 | |
| 		nc->current_recursive = nh;
 | |
| 	} else
 | |
| 		start_recursive_chain(nc, nh);
 | |
| 
 | |
| 	add_string_representation(&nc->repr, nh);
 | |
| }
 | |
| 
 | |
| static void nexthop_clear_recursive(struct nexthop *tcur)
 | |
| {
 | |
| 	if (!tcur)
 | |
| 		return;
 | |
| 	if (CHECK_FLAG(tcur->flags, NEXTHOP_FLAG_RECURSIVE))
 | |
| 		nexthop_clear_recursive(tcur->resolved);
 | |
| 	if (tcur->next)
 | |
| 		nexthop_clear_recursive(tcur->next);
 | |
| 	free(tcur);
 | |
| }
 | |
| static void nexthop_chain_clear(struct nexthop_chain *nc)
 | |
| {
 | |
| 	nexthop_clear_recursive(nc->head.nexthop);
 | |
| 	nc->head.nexthop = nc->current_top = nc->current_recursive = NULL;
 | |
| 	free(nc->repr);
 | |
| 	nc->repr = NULL;
 | |
| }
 | |
| 
 | |
| static void nexthop_chain_free(struct nexthop_chain *nc)
 | |
| {
 | |
| 	if (!nc)
 | |
| 		return;
 | |
| 	nexthop_chain_clear(nc);
 | |
| 	free(nc);
 | |
| }
 | |
| 
 | |
| /* This function builds a string representation of
 | |
|  * the nexthop chain using the ALL_NEXTHOPS macro.
 | |
|  * It verifies that the ALL_NEXTHOPS macro iterated
 | |
|  * correctly over the nexthop chain by comparing the
 | |
|  * generated representation with the expected representation.
 | |
|  */
 | |
| static void nexthop_chain_verify_iter(struct nexthop_chain *nc)
 | |
| {
 | |
| 	struct nexthop *nh;
 | |
| 	char *repr = NULL;
 | |
| 
 | |
| 	for (ALL_NEXTHOPS(nc->head, nh))
 | |
| 		add_string_representation(&repr, nh);
 | |
| 
 | |
| 	if (repr && verbose)
 | |
| 		printf("===\n%s", repr);
 | |
| 	assert((!repr && !nc->repr)
 | |
| 	       || (repr && nc->repr && !strcmp(repr, nc->repr)));
 | |
| 	free(repr);
 | |
| }
 | |
| 
 | |
| /* This test run builds a simple nexthop chain
 | |
|  * with some recursive nexthops and verifies that
 | |
|  * the iterator works correctly in each stage along
 | |
|  * the way.
 | |
|  */
 | |
| static void test_run_first(void)
 | |
| {
 | |
| 	struct nexthop_chain *nc;
 | |
| 
 | |
| 	nc = nexthop_chain_new();
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_top(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_top(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_recursive(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_recursive(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_top(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_top(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_top(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_recursive(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_recursive(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_recursive(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_recursive_level(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_recursive(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_recursive(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_add_top(nc);
 | |
| 	nexthop_chain_verify_iter(nc);
 | |
| 
 | |
| 	nexthop_chain_free(nc);
 | |
| }
 | |
| 
 | |
| /* This test run builds numerous random
 | |
|  * nexthop chain configurations and verifies
 | |
|  * that the iterator correctly progresses
 | |
|  * through each. */
 | |
| static void test_run_prng(void)
 | |
| {
 | |
| 	struct nexthop_chain *nc;
 | |
| 	struct prng *prng;
 | |
| 	int i;
 | |
| 
 | |
| 	nc = nexthop_chain_new();
 | |
| 	prng = prng_new(0);
 | |
| 
 | |
| 	for (i = 0; i < 1000000; i++) {
 | |
| 		switch (prng_rand(prng) % 10) {
 | |
| 		case 0:
 | |
| 			nexthop_chain_clear(nc);
 | |
| 			break;
 | |
| 		case 1:
 | |
| 		case 2:
 | |
| 		case 3:
 | |
| 		case 4:
 | |
| 		case 5:
 | |
| 			nexthop_chain_add_top(nc);
 | |
| 			break;
 | |
| 		case 6:
 | |
| 		case 7:
 | |
| 		case 8:
 | |
| 			if (nc->current_top)
 | |
| 				nexthop_chain_add_recursive(nc);
 | |
| 			break;
 | |
| 		case 9:
 | |
| 			if (nc->current_top)
 | |
| 				nexthop_chain_add_recursive_level(nc);
 | |
| 			break;
 | |
| 		}
 | |
| 		nexthop_chain_verify_iter(nc);
 | |
| 	}
 | |
| 	nexthop_chain_free(nc);
 | |
| 	prng_free(prng);
 | |
| }
 | |
| 
 | |
| int main(int argc, char **argv)
 | |
| {
 | |
| 	if (argc >= 2 && !strcmp("-v", argv[1]))
 | |
| 		verbose = 1;
 | |
| 	test_run_first();
 | |
| 	printf("Simple test passed.\n");
 | |
| 	test_run_prng();
 | |
| 	printf("PRNG test passed.\n");
 | |
| }
 |