diff options
author | Ruediger Pluem <rpluem@apache.org> | 2006-04-14 15:20:28 +0200 |
---|---|---|
committer | Ruediger Pluem <rpluem@apache.org> | 2006-04-14 15:20:28 +0200 |
commit | 96c6356987991704cbf490733c21d4d605f27b9a (patch) | |
tree | 69d35422817af00df6c14d41d82a18e928d4d5a3 /modules/proxy/proxy_util.c | |
parent | * Initialize last_char as otherwise a random value will be compared (diff) | |
download | apache2-96c6356987991704cbf490733c21d4d605f27b9a.tar.xz apache2-96c6356987991704cbf490733c21d4d605f27b9a.zip |
* Avoid calling ap_proxy_http_cleanup twice as this releases a connection
from the connection pool twice. This causes this connection to be present
in the connection pool twice. Thus it may be used by different threads
at the same time which causes many troubles (segfaults in this case).
Furthermore implement a logic to prevent double releases to the connection
pool if they are triggered by buggy code and log an error message in this
case.
- mod_proxy_http.c: remove double calls to ap_proxy_http_cleanup
- proxy_util.c: Add logic to prevent double releases of a
connection to the connection pool.
PR: 38793
git-svn-id: https://svn.apache.org/repos/asf/httpd/httpd/trunk@394088 13f79535-47bb-0310-9956-ffa450edef68
Diffstat (limited to 'modules/proxy/proxy_util.c')
-rw-r--r-- | modules/proxy/proxy_util.c | 16 |
1 files changed, 15 insertions, 1 deletions
diff --git a/modules/proxy/proxy_util.c b/modules/proxy/proxy_util.c index 104b465e68..a63bfcaf63 100644 --- a/modules/proxy/proxy_util.c +++ b/modules/proxy/proxy_util.c @@ -1512,7 +1512,18 @@ static apr_status_t connection_cleanup(void *theconn) if (!worker->cp) return APR_SUCCESS; - /* deterimine if the connection need to be closed */ +#if APR_HAS_THREADS + /* Sanity check: Did we already return the pooled connection? */ + if (conn->inreslist) { + ap_log_perror(APLOG_MARK, APLOG_ERR, 0, conn->pool, + "proxy: Pooled connection 0x%pp for worker %s has been" + " already returned to the connection pool.", conn, + worker->name); + return APR_SUCCESS; + } +#endif + + /* determine if the connection need to be closed */ if (conn->close_on_recycle || conn->close) { apr_pool_t *p = conn->pool; apr_pool_clear(conn->pool); @@ -1522,6 +1533,7 @@ static apr_status_t connection_cleanup(void *theconn) } #if APR_HAS_THREADS if (worker->hmax && worker->cp->res) { + conn->inreslist = 1; apr_reslist_release(worker->cp->res, (void *)conn); } else @@ -1552,6 +1564,7 @@ static apr_status_t connection_constructor(void **resource, void *params, conn->pool = ctx; conn->worker = worker; + conn->inreslist = 1; *resource = conn; return APR_SUCCESS; @@ -1784,6 +1797,7 @@ PROXY_DECLARE(int) ap_proxy_acquire_connection(const char *proxy_function, (*conn)->worker = worker; (*conn)->close = 0; (*conn)->close_on_recycle = 0; + (*conn)->inreslist = 0; return OK; } |