mirror of
				https://github.com/openvswitch/ovs
				synced 2025-10-23 14:57:06 +00:00 
			
		
		
		
	
		
			
	
	
		
			241 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
		
		
			
		
	
	
			241 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
|   | /*
 | ||
|  |  * Copyright (c) 2009 Nicira Networks. | ||
|  |  * | ||
|  |  * Licensed under the Apache License, Version 2.0 (the "License"); | ||
|  |  * you may not use this file except in compliance with the License. | ||
|  |  * You may obtain a copy of the License at: | ||
|  |  * | ||
|  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | ||
|  |  * | ||
|  |  * Unless required by applicable law or agreed to in writing, software | ||
|  |  * distributed under the License is distributed on an "AS IS" BASIS, | ||
|  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||
|  |  * See the License for the specific language governing permissions and | ||
|  |  * limitations under the License. | ||
|  |  */ | ||
|  | 
 | ||
|  | #include <config.h>
 | ||
|  | 
 | ||
|  | #include "reconnect.h"
 | ||
|  | 
 | ||
|  | #include <errno.h>
 | ||
|  | #include <stdio.h>
 | ||
|  | #include <stdlib.h>
 | ||
|  | #include <string.h>
 | ||
|  | 
 | ||
|  | #include "command-line.h"
 | ||
|  | #include "compiler.h"
 | ||
|  | #include "svec.h"
 | ||
|  | #include "util.h"
 | ||
|  | 
 | ||
|  | static struct reconnect *reconnect; | ||
|  | static int now; | ||
|  | 
 | ||
|  | static const struct command commands[]; | ||
|  | 
 | ||
|  | static void diff_stats(const struct reconnect_stats *old, | ||
|  |                        const struct reconnect_stats *new); | ||
|  | 
 | ||
|  | int | ||
|  | main(void) | ||
|  | { | ||
|  |     struct reconnect_stats prev; | ||
|  |     int old_time; | ||
|  |     char line[128]; | ||
|  | 
 | ||
|  |     now = 1000; | ||
|  |     reconnect = reconnect_create(now); | ||
|  |     reconnect_set_name(reconnect, "remote"); | ||
|  |     reconnect_get_stats(reconnect, now, &prev); | ||
|  |     printf("### t=%d ###\n", now); | ||
|  |     old_time = now; | ||
|  |     while (fgets(line, sizeof line, stdin)) { | ||
|  |         struct reconnect_stats cur; | ||
|  |         struct svec args; | ||
|  | 
 | ||
|  |         fputs(line, stdout); | ||
|  |         if (line[0] == '#') { | ||
|  |             continue; | ||
|  |         } | ||
|  | 
 | ||
|  |         svec_init(&args); | ||
|  |         svec_parse_words(&args, line); | ||
|  |         svec_terminate(&args); | ||
|  |         if (!svec_is_empty(&args)) { | ||
|  |             run_command(args.n, args.names, commands); | ||
|  |         } | ||
|  |         svec_destroy(&args); | ||
|  | 
 | ||
|  |         if (old_time != now) { | ||
|  |             printf("\n### t=%d ###\n", now); | ||
|  |             old_time = now; | ||
|  |         } | ||
|  | 
 | ||
|  |         reconnect_get_stats(reconnect, now, &cur); | ||
|  |         diff_stats(&prev, &cur); | ||
|  |         prev = cur; | ||
|  |     } | ||
|  | 
 | ||
|  |     return 0; | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_enable(int argc UNUSED, char *argv[] UNUSED) | ||
|  | { | ||
|  |     reconnect_enable(reconnect, now); | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_disable(int argc UNUSED, char *argv[] UNUSED) | ||
|  | { | ||
|  |     reconnect_disable(reconnect, now); | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_force_reconnect(int argc UNUSED, char *argv[] UNUSED) | ||
|  | { | ||
|  |     reconnect_force_reconnect(reconnect, now); | ||
|  | } | ||
|  | 
 | ||
|  | static int | ||
|  | error_from_string(const char *s) | ||
|  | { | ||
|  |     if (!s) { | ||
|  |         return 0; | ||
|  |     } else if (!strcmp(s, "ECONNREFUSED")) { | ||
|  |         return ECONNREFUSED; | ||
|  |     } else if (!strcmp(s, "EOF")) { | ||
|  |         return EOF; | ||
|  |     } else { | ||
|  |         ovs_fatal(0, "unknown error '%s'", s); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_disconnected(int argc UNUSED, char *argv[]) | ||
|  | { | ||
|  |     reconnect_disconnected(reconnect, now, error_from_string(argv[1])); | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_connecting(int argc UNUSED, char *argv[] UNUSED) | ||
|  | { | ||
|  |     reconnect_connecting(reconnect, now); | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_connect_failed(int argc UNUSED, char *argv[]) | ||
|  | { | ||
|  |     reconnect_connect_failed(reconnect, now, error_from_string(argv[1])); | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_connected(int argc UNUSED, char *argv[] UNUSED) | ||
|  | { | ||
|  |     reconnect_connected(reconnect, now); | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_received(int argc UNUSED, char *argv[] UNUSED) | ||
|  | { | ||
|  |     reconnect_received(reconnect, now); | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_run(int argc, char *argv[]) | ||
|  | { | ||
|  |     enum reconnect_action action; | ||
|  | 
 | ||
|  |     if (argc > 1) { | ||
|  |         now += atoi(argv[1]); | ||
|  |     } | ||
|  | 
 | ||
|  |     action = reconnect_run(reconnect, now); | ||
|  |     switch (action) { | ||
|  |     default: | ||
|  |         if (action != 0) { | ||
|  |             NOT_REACHED(); | ||
|  |         } | ||
|  |         break; | ||
|  | 
 | ||
|  |     case RECONNECT_CONNECT: | ||
|  |         printf("  should connect\n"); | ||
|  |         break; | ||
|  | 
 | ||
|  |     case RECONNECT_DISCONNECT: | ||
|  |         printf("  should disconnect\n"); | ||
|  |         break; | ||
|  | 
 | ||
|  |     case RECONNECT_PROBE: | ||
|  |         printf("  should send probe\n"); | ||
|  |         break; | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_advance(int argc UNUSED, char *argv[]) | ||
|  | { | ||
|  |     now += atoi(argv[1]); | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | do_timeout(int argc UNUSED, char *argv[] UNUSED) | ||
|  | { | ||
|  |     int timeout = reconnect_timeout(reconnect, now); | ||
|  |     if (timeout >= 0) { | ||
|  |         printf("  advance %d ms\n", timeout); | ||
|  |         now += timeout; | ||
|  |     } else { | ||
|  |         printf("  no timeout\n"); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | static void | ||
|  | diff_stats(const struct reconnect_stats *old, | ||
|  |            const struct reconnect_stats *new) | ||
|  | { | ||
|  |     if (old->state != new->state | ||
|  |         || old->state_elapsed != new->state_elapsed | ||
|  |         || old->backoff != new->backoff) { | ||
|  |         printf("  in %s for %u ms (%d ms backoff)\n", | ||
|  |                new->state, new->state_elapsed, new->backoff); | ||
|  |     } | ||
|  |     if (old->creation_time != new->creation_time | ||
|  |         || old->last_received != new->last_received | ||
|  |         || old->last_connected != new->last_connected) { | ||
|  |         printf("  created %lld, last received %lld, last connected %lld\n", | ||
|  |                new->creation_time, new->last_received, new->last_connected); | ||
|  |     } | ||
|  |     if (old->n_successful_connections != new->n_successful_connections | ||
|  |         || old->n_attempted_connections != new->n_attempted_connections | ||
|  |         || old->seqno != new->seqno) { | ||
|  |         printf("  %u successful connections out of %u attempts, seqno %u\n", | ||
|  |                new->n_successful_connections, new->n_attempted_connections, | ||
|  |                new->seqno); | ||
|  |     } | ||
|  |     if (old->is_connected != new->is_connected | ||
|  |         || old->current_connection_duration != new->current_connection_duration | ||
|  |         || old->total_connected_duration != new->total_connected_duration) { | ||
|  |         printf("  %sconnected (%u ms), total %u ms connected\n", | ||
|  |                new->is_connected ? "" : "not ", | ||
|  |                new->current_connection_duration, | ||
|  |                new->total_connected_duration); | ||
|  |     } | ||
|  | } | ||
|  | 
 | ||
|  | static const struct command commands[] = { | ||
|  |     { "enable", 0, 0, do_enable }, | ||
|  |     { "disable", 0, 0, do_disable }, | ||
|  |     { "force-reconnect", 0, 0, do_force_reconnect }, | ||
|  |     { "disconnected", 0, 1, do_disconnected }, | ||
|  |     { "connecting", 0, 0, do_connecting }, | ||
|  |     { "connect-failed", 0, 1, do_connect_failed }, | ||
|  |     { "connected", 0, 0, do_connected }, | ||
|  |     { "received", 0, 0, do_received }, | ||
|  |     { "run", 0, 1, do_run }, | ||
|  |     { "advance", 1, 1, do_advance }, | ||
|  |     { "timeout", 0, 0, do_timeout }, | ||
|  |     { NULL, 0, 0, NULL }, | ||
|  | }; | ||
|  | 
 |