summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorNick Mathewson <nickm@torproject.org>2008-02-20 22:28:26 +0000
committerNick Mathewson <nickm@torproject.org>2008-02-20 22:28:26 +0000
commit93aa3355161923d722d196ebb27f35c8138c4c69 (patch)
tree8790b4ceddf2e412d39972a86817226d2e658806
parent644452a62fde5fec66dab56c4f47204f42b57bac (diff)
downloadtor-93aa3355161923d722d196ebb27f35c8138c4c69.tar.gz
tor-93aa3355161923d722d196ebb27f35c8138c4c69.zip
r18269@catbus: nickm | 2008-02-20 17:28:24 -0500
Apply patch from Sebastian Hahn: stop imposing an arbitrary maximum on the number of file descriptors used for busy servers. Bug reported by Olaf Selke. svn:r13626
-rw-r--r--ChangeLog3
-rw-r--r--src/common/compat.c75
-rw-r--r--src/common/compat.h2
-rw-r--r--src/or/config.c8
-rw-r--r--src/or/or.h12
5 files changed, 62 insertions, 38 deletions
diff --git a/ChangeLog b/ChangeLog
index ae981ade36..ec4ab45ca4 100644
--- a/ChangeLog
+++ b/ChangeLog
@@ -30,6 +30,9 @@ Changes in version 0.2.0.20-?? - 2008-02-??
- Add a --with-tcmalloc option to the configure script to link
against tcmalloc (if present). Does not yet search for
non-system include paths.
+ - Stop imposing an arbitrary maximum on the number of file descriptors
+ used for busy servers. Bug reported by Olaf Selke; patch from
+ Sebastian Hahn.
o Minor features (controller):
- Add a new __HashedControlSessionPassword option for controllers
diff --git a/src/common/compat.c b/src/common/compat.c
index d443fc6aa6..cb88864f27 100644
--- a/src/common/compat.c
+++ b/src/common/compat.c
@@ -756,46 +756,76 @@ typedef unsigned long rlim_t;
/** Learn the maximum allowed number of file descriptors. (Some systems
* have a low soft limit.
*
- * We compute this by finding the largest number between <b>limit</b>
- * and <b>cap</b> that we can use. If we can't find a number greater
- * than or equal to <b>limit</b>, then we fail: return -1.
+ * We compute this by finding the largest number that we can use.
+ * If we can't find a number greater than or equal to <b>limit</b>,
+ * then we fail: return -1.
*
- * Otherwise, return the number minus some buffer to allow for other
- * file descriptors we'll want available for ordinary use. */
+ * Otherwise, return 0 and store the maximum we found inside <b>max_out</b>.*/
int
-set_max_file_descriptors(unsigned long limit, unsigned long cap)
-{
+set_max_file_descriptors(unsigned long limit, int *max_out)
+{
+#define DEFAULT_MAX_CONNECTIONS 15000
+#define CYGWIN_MAX_CONNECTIONS 3200
+#define IPHONE_MAX_CONNECTIONS 9999
+ /* Define some maximum connections values for systems where we cannot
+ * automatically determine a limit. Re Cygwin, see
+ * http://archives.seul.org/or/talk/Aug-2006/msg00210.html
+ * For an iPhone, 9999 should work. For Windows and all other unknown
+ * systems we use 15000 as the default. */
#ifndef HAVE_GETRLIMIT
log_fn(LOG_INFO, LD_NET,
"This platform is missing getrlimit(). Proceeding.");
- if (limit < cap) {
- log_info(LD_CONFIG, "ConnLimit must be at most %d. Using that.", (int)cap);
- limit = cap;
+#ifdef MS_WINDOWS
+ if (limit > DEFAULT_MAX_CONNECTIONS) {
+ log_warn(LD_CONFIG,
+ "We do not support more than %lu file descriptors",
+ "on Windows. Tried to raise to %lu.",
+ DEFAULT_MAX_CONNECTIONS, limit);
+ return -1;
}
+ limit = DEFAULT_MAX_CONNECTIONS;
+#elif defined(CYGWIN) || defined(__CYGWIN__)
+ if (limit > CYGWIN_MAX_CONNECTIONS) {
+ log_warn(LD_CONFIG, "We do not support more than %lu file descriptors",
+ "when using Cygwin. Tried to raise to %lu.",
+ CYGWIN_MAX_CONNECTIONS, limit);
+ return -1;
+ }
+ limit = CYGWIN_MAX_CONNECTIONS;
+#elif defined(IPHONE)
+ if (limit > IPHONE_MAX_CONNECTIONS) {
+ log_warn(LD_CONFIG, "We do not support more than %lu file descriptors",
+ "on iPhone. Tried to raise to %lu.",
+ IPHONE_MAX_CONNECTIONS, limit);
+ return -1;
+ }
+ limit = IPHONE_MAX_CONNECTIONS;
+#else
+ /* Unknown system without getrlimit support. Use the default value.*/
+ limit = DEFAULT_MAX_CONNECTIONS;
+#endif
#else
struct rlimit rlim;
- rlim_t most;
tor_assert(limit > 0);
- tor_assert(cap > 0);
if (getrlimit(RLIMIT_NOFILE, &rlim) != 0) {
log_warn(LD_NET, "Could not get maximum number of file descriptors: %s",
strerror(errno));
return -1;
}
- //log_notice(LD_CONFIG, "%llu %llu", rlim.rlim_cur, rlim.rlim_max);
+
if ((unsigned long)rlim.rlim_max < limit) {
log_warn(LD_CONFIG,"We need %lu file descriptors available, and we're "
"limited to %lu. Please change your ulimit -n.",
limit, (unsigned long)rlim.rlim_max);
return -1;
}
- most = rlim.rlim_max > (rlim_t)cap ? (rlim_t)cap : rlim.rlim_max;
- if ((rlim_t)most > rlim.rlim_cur) {
+
+ if (rlim.rlim_max > rlim.rlim_cur) {
log_info(LD_NET,"Raising max file descriptors from %lu to %lu.",
- (unsigned long)rlim.rlim_cur, (unsigned long)most);
+ (unsigned long)rlim.rlim_cur, (unsigned long)rlim.rlim_max);
}
- rlim.rlim_cur = most;
+ rlim.rlim_cur = rlim.rlim_max;
if (setrlimit(RLIMIT_NOFILE, &rlim) != 0) {
int bad = 1;
@@ -812,9 +842,8 @@ set_max_file_descriptors(unsigned long limit, unsigned long cap)
} else {
log_info(LD_CONFIG, "Dropped connection limit to OPEN_MAX (%lu); "
"Apparently, %lu was too high and rlimit lied to us.",
- (unsigned long)OPEN_MAX, (unsigned long)most);
+ (unsigned long)OPEN_MAX, (unsigned long)rlim.rlim_max);
}
- most = rlim.rlim_cur;
bad = 0;
}
}
@@ -826,7 +855,7 @@ set_max_file_descriptors(unsigned long limit, unsigned long cap)
}
}
/* leave some overhead for logs, etc, */
- limit = most;
+ limit = rlim.rlim_cur;
#endif
if (limit < ULIMIT_BUFFER) {
@@ -834,7 +863,11 @@ set_max_file_descriptors(unsigned long limit, unsigned long cap)
"ConnLimit must be at least %d. Failing.", ULIMIT_BUFFER);
return -1;
}
- return limit - ULIMIT_BUFFER;
+ if (limit > INT_MAX)
+ limit = INT_MAX;
+ tor_assert(max_out);
+ *max_out = (int)limit - ULIMIT_BUFFER;
+ return 0;
}
/** Call setuid and setgid to run as <b>user</b>:<b>group</b>. Return 0 on
diff --git a/src/common/compat.h b/src/common/compat.h
index 7be7fddcdd..0d6fd87dde 100644
--- a/src/common/compat.h
+++ b/src/common/compat.h
@@ -442,7 +442,7 @@ uint32_t get_uint32(const char *cp) ATTR_PURE ATTR_NONNULL((1));
void set_uint16(char *cp, uint16_t v) ATTR_NONNULL((1));
void set_uint32(char *cp, uint32_t v) ATTR_NONNULL((1));
-int set_max_file_descriptors(unsigned long limit, unsigned long cap);
+int set_max_file_descriptors(unsigned long limit, int *max);
int switch_id(const char *user, const char *group);
#ifdef HAVE_PWD_H
char *get_user_homedir(const char *username);
diff --git a/src/or/config.c b/src/or/config.c
index 5781682c12..5903d25b15 100644
--- a/src/or/config.c
+++ b/src/or/config.c
@@ -990,9 +990,8 @@ options_act_reversible(or_options_t *old_options, char **msg)
if (running_tor) {
/* We need to set the connection limit before we can open the listeners. */
- options->_ConnLimit =
- set_max_file_descriptors((unsigned)options->ConnLimit, MAXCONNECTIONS);
- if (options->_ConnLimit < 0) {
+ if (set_max_file_descriptors((unsigned)options->ConnLimit,
+ &options->_ConnLimit) < 0) {
*msg = tor_strdup("Problem with ConnLimit value. See logs for details.");
goto rollback;
}
@@ -1075,7 +1074,8 @@ options_act_reversible(or_options_t *old_options, char **msg)
}
if (set_conn_limit && old_options)
- set_max_file_descriptors((unsigned)old_options->ConnLimit,MAXCONNECTIONS);
+ set_max_file_descriptors((unsigned)old_options->ConnLimit,
+ &options->_ConnLimit);
SMARTLIST_FOREACH(new_listeners, connection_t *, conn,
{
diff --git a/src/or/or.h b/src/or/or.h
index f4427089c3..50d98e82ab 100644
--- a/src/or/or.h
+++ b/src/or/or.h
@@ -65,18 +65,6 @@
#include <time.h>
#endif
-/** Upper bound on maximum simultaneous connections; can be lowered by
- * config file. */
-#if defined(CYGWIN) || defined(__CYGWIN__) || defined(IPHONE)
-/* Re Cygwin, see http://archives.seul.org/or/talk/Aug-2006/msg00210.html */
-/* For an iPhone, the limit would be closer to 9999. But nobody seems to be
- * running a server on an iPhone anyway? */
-#define MAXCONNECTIONS 3200
-#else
-/* very high by default. "nobody should need more than this..." */
-#define MAXCONNECTIONS 15000
-#endif
-
#ifdef MS_WINDOWS
#include <io.h>
#include <process.h>