# HG changeset patch # User Matt Johnston # Date 1222097332 0 # Node ID 6cd2152aae0b6dd3803d4efb7cf736b8ac8dca2a # Parent b956d6151600807c11d0b1dce60ba2e4760c994b Idle-timeout patch from Farrell Aultman, need to figure whether to only account DATA packets and whether server->client data makes sense too. diff -r b956d6151600 -r 6cd2152aae0b cli-runopts.c --- a/cli-runopts.c Mon Sep 22 14:13:44 2008 +0000 +++ b/cli-runopts.c Mon Sep 22 15:28:52 2008 +0000 @@ -70,6 +70,7 @@ #endif "-W (default %d, larger may be faster, max 1MB)\n" "-K (0 is never, default %d)\n" + "-I (0 is never, default %d)\n" #ifdef ENABLE_CLI_NETCAT "-B Netcat-alike bouncing\n" #endif @@ -80,7 +81,7 @@ "-v verbose\n" #endif ,DROPBEAR_VERSION, cli_opts.progname, - DEFAULT_RECV_WINDOW, DEFAULT_KEEPALIVE); + DEFAULT_RECV_WINDOW, DEFAULT_KEEPALIVE, DEFAULT_IDLE_TIMEOUT); } @@ -105,6 +106,7 @@ char* recv_window_arg = NULL; char* keepalive_arg = NULL; + char* idle_timeout_arg = NULL; /* see printhelp() for options */ cli_opts.progname = argv[0]; @@ -252,6 +254,9 @@ case 'K': next = &keepalive_arg; break; + case 'I': + next = &idle_timeout_arg; + break; #ifdef DEBUG_TRACE case 'v': debug_trace = 1; @@ -359,6 +364,11 @@ dropbear_exit("Bad keepalive '%s'", keepalive_arg); } } + if (idle_timeout_arg) { + if (m_str_to_uint(idle_timeout_arg, &opts.idle_timeout_secs) == DROPBEAR_FAILURE) { + dropbear_exit("Bad idle_timeout '%s'", idle_timeout_arg); + } + } #ifdef ENABLE_CLI_NETCAT if (cli_opts.cmd && cli_opts.netcat_host) { diff -r b956d6151600 -r 6cd2152aae0b common-session.c --- a/common-session.c Mon Sep 22 14:13:44 2008 +0000 +++ b/common-session.c Mon Sep 22 15:28:52 2008 +0000 @@ -64,6 +64,7 @@ ses.connect_time = 0; ses.last_packet_time = 0; + ses.last_recv_packet_time = 0; if (pipe(ses.signal_pipe) < 0) { dropbear_exit("signal pipe failed"); @@ -256,7 +257,7 @@ ses.remoteclosed(); } - /* If they send more than 50 lines, something is wrong */ + /* If they send more than 50 lines, something is wrong */ for (i = 0; i < 50; i++) { len = ident_readln(ses.sock_in, linebuf, sizeof(linebuf)); @@ -281,11 +282,11 @@ memcpy(ses.remoteident, linebuf, len); } - /* Shall assume that 2.x will be backwards compatible. */ - if (strncmp(ses.remoteident, "SSH-2.", 6) != 0 - && strncmp(ses.remoteident, "SSH-1.99-", 9) != 0) { - dropbear_exit("Incompatible remote version '%s'", ses.remoteident); - } + /* Shall assume that 2.x will be backwards compatible. */ + if (strncmp(ses.remoteident, "SSH-2.", 6) != 0 + && strncmp(ses.remoteident, "SSH-1.99-", 9) != 0) { + dropbear_exit("Incompatible remote version '%s'", ses.remoteident); + } TRACE(("remoteident: %s", ses.remoteident)) @@ -400,6 +401,11 @@ && now - ses.last_packet_time >= opts.keepalive_secs) { send_msg_ignore(); } + + if (opts.idle_timeout_secs > 0 && ses.last_recv_packet_time > 0 + && now - ses.last_recv_packet_time >= opts.idle_timeout_secs) { + dropbear_close("Idle timeout"); + } } static long select_timeout() { @@ -412,6 +418,8 @@ ret = MIN(AUTH_TIMEOUT, ret); if (opts.keepalive_secs > 0) ret = MIN(opts.keepalive_secs, ret); + if (opts.idle_timeout_secs > 0) + ret = MIN(opts.idle_timeout_secs, ret); return ret; } diff -r b956d6151600 -r 6cd2152aae0b options.h --- a/options.h Mon Sep 22 14:13:44 2008 +0000 +++ b/options.h Mon Sep 22 15:28:52 2008 +0000 @@ -243,6 +243,10 @@ be overridden at runtime with -K. 0 disables keepalives */ #define DEFAULT_KEEPALIVE 0 +/* Ensure that data is received within IDLE_TIMEOUT seconds. This can +be overridden at runtime with -I. 0 disables idle timeouts */ +#define DEFAULT_IDLE_TIMEOUT 0 + /******************************************************************* * You shouldn't edit below here unless you know you need to. *******************************************************************/ diff -r b956d6151600 -r 6cd2152aae0b process-packet.c --- a/process-packet.c Mon Sep 22 14:13:44 2008 +0000 +++ b/process-packet.c Mon Sep 22 15:28:52 2008 +0000 @@ -70,6 +70,7 @@ dropbear_close("Disconnect received"); } + ses.last_recv_packet_time = time(NULL); /* This applies for KEX, where the spec says the next packet MUST be * NEWKEYS */ diff -r b956d6151600 -r 6cd2152aae0b runopts.h --- a/runopts.h Mon Sep 22 14:13:44 2008 +0000 +++ b/runopts.h Mon Sep 22 15:28:52 2008 +0000 @@ -38,7 +38,7 @@ #endif unsigned int recv_window; unsigned int keepalive_secs; - + unsigned int idle_timeout_secs; } runopts; extern runopts opts; diff -r b956d6151600 -r 6cd2152aae0b session.h --- a/session.h Mon Sep 22 14:13:44 2008 +0000 +++ b/session.h Mon Sep 22 15:28:52 2008 +0000 @@ -134,12 +134,16 @@ unsigned char lastpacket; /* What the last received packet type was */ - int signal_pipe[2]; /* stores endpoints of a self-pipe used for + int signal_pipe[2]; /* stores endpoints of a self-pipe used for race-free signal handling */ time_t last_packet_time; /* time of the last packet transmission, for keepalive purposes */ + time_t last_recv_packet_time; /* time of the last packet received, for + idle timeout purposes */ + + /* KEX/encryption related */ struct KEXState kexstate; struct key_context *keys; diff -r b956d6151600 -r 6cd2152aae0b svr-runopts.c --- a/svr-runopts.c Mon Sep 22 14:13:44 2008 +0000 +++ b/svr-runopts.c Mon Sep 22 15:28:52 2008 +0000 @@ -82,6 +82,7 @@ #endif "-W (default %d, larger may be faster, max 1MB)\n" "-K (0 is never, default %d)\n" + "-I (0 is never, default %d)\n" #ifdef DEBUG_TRACE "-v verbose\n" #endif @@ -93,7 +94,7 @@ RSA_PRIV_FILENAME, #endif DROPBEAR_MAX_PORTS, DROPBEAR_DEFPORT, DROPBEAR_PIDFILE, - DEFAULT_RECV_WINDOW, DEFAULT_KEEPALIVE); + DEFAULT_RECV_WINDOW, DEFAULT_KEEPALIVE, DEFAULT_IDLE_TIMEOUT); } void svr_getopts(int argc, char ** argv) { @@ -103,6 +104,7 @@ int nextisport = 0; char* recv_window_arg = NULL; char* keepalive_arg = NULL; + char* idle_timeout_arg = NULL; /* see printhelp() for options */ svr_opts.rsakeyfile = NULL; @@ -134,7 +136,8 @@ svr_opts.usingsyslog = 1; #endif opts.recv_window = DEFAULT_RECV_WINDOW; - opts.keepalive_secs = DEFAULT_KEEPALIVE; + opts.keepalive_secs = DEFAULT_KEEPALIVE; + opts.idle_timeout_secs = DEFAULT_IDLE_TIMEOUT; #ifdef ENABLE_SVR_REMOTETCPFWD opts.listen_fwd_all = 0; @@ -218,6 +221,9 @@ case 'K': next = &keepalive_arg; break; + case 'I': + next = &idle_timeout_arg; + break; #if defined(ENABLE_SVR_PASSWORD_AUTH) || defined(ENABLE_SVR_PAM_AUTH) case 's': svr_opts.noauthpass = 1; @@ -253,7 +259,7 @@ svr_opts.addresses[0] = m_strdup(DROPBEAR_DEFADDRESS); svr_opts.portcount = 1; } - + if (svr_opts.dsskeyfile == NULL) { svr_opts.dsskeyfile = DSS_PRIV_FILENAME; } @@ -294,6 +300,13 @@ dropbear_exit("Bad keepalive '%s'", keepalive_arg); } } + + if (idle_timeout_arg) { + opts.idle_timeout_secs = strtoul(idle_timeout_arg, NULL, 10); + if (opts.idle_timeout_secs == 0 && errno == EINVAL) { + dropbear_exit("Bad idle_timeout '%s'", idle_timeout_arg); + } + } } static void addportandaddress(char* spec) {