}__attribute__((packed));
+/* This structure is used to manage idle connections, their locking, and the
+ * list of such idle connections to be removed. It is per-thread and must be
+ * accessible from foreign threads.
+ */
+struct idle_conns {
+ struct mt_list toremove_conns;
+ __decl_thread(HA_SPINLOCK_T toremove_lock);
+ struct task *cleanup_task;
+} THREAD_ALIGNED(64);
+
#endif /* _HAPROXY_CONNECTION_T_H */
/*
/* If we delayed the mux creation because we were waiting for the handshake, do it now */
int conn_create_mux(struct connection *conn);
-__decl_thread(extern HA_SPINLOCK_T toremove_lock[MAX_THREADS]);
+extern struct idle_conns idle_conns[MAX_THREADS];
/* returns true is the transport layer is ready */
static inline int conn_xprt_ready(const struct connection *conn)
}
conn_force_unsubscribe(conn);
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
MT_LIST_DEL((struct mt_list *)&conn->list);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
pool_free(pool_head_connection, conn);
}
__decl_thread(extern HA_SPINLOCK_T idle_conn_srv_lock);
extern struct eb_root idle_conn_srv;
extern struct task *idle_conn_task;
-extern struct task *idle_conn_cleanup[MAX_THREADS];
-extern struct mt_list toremove_connections[MAX_THREADS];
extern struct dict server_name_dict;
int srv_downtime(const struct server *s);
* thread may be trying to migrate that connection, and we don't want
* to end up with two threads using the same connection.
*/
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
conn = MT_LIST_POP(&mt_list[tid], struct connection *, list);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
/* If we found a connection in our own list, and we don't have to
* steal one from another thread, then we're done.
for (i = tid; !found && (i = ((i + 1 == global.nbthread) ? 0 : i + 1)) != tid;) {
struct mt_list *elt1, elt2;
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[i]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[i].toremove_lock);
mt_list_for_each_entry_safe(conn, &mt_list[i], list, elt1, elt2) {
if (conn->mux->takeover && conn->mux->takeover(conn) == 0) {
MT_LIST_DEL_SAFE(elt1);
break;
}
}
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[i]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[i].toremove_lock);
}
if (!found)
// see it possibly larger.
ALREADY_CHECKED(i);
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
tokill_conn = MT_LIST_POP(&srv->idle_conns[i],
struct connection *, list);
if (!tokill_conn)
if (tokill_conn) {
/* We got one, put it into the concerned thread's to kill list, and wake it's kill task */
- MT_LIST_ADDQ(&toremove_connections[i],
+ MT_LIST_ADDQ(&idle_conns[i].toremove_conns,
(struct mt_list *)&tokill_conn->list);
- task_wakeup(idle_conn_cleanup[i], TASK_WOKEN_OTHER);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ task_wakeup(idle_conns[i].cleanup_task, TASK_WOKEN_OTHER);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
break;
}
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
}
}
idle_conn_task->context = NULL;
for (i = 0; i < global.nbthread; i++) {
- idle_conn_cleanup[i] = task_new(1UL << i);
- if (!idle_conn_cleanup[i])
+ idle_conns[i].cleanup_task = task_new(1UL << i);
+ if (!idle_conns[i].cleanup_task)
goto err;
- idle_conn_cleanup[i]->process = srv_cleanup_toremove_connections;
- idle_conn_cleanup[i]->context = NULL;
- MT_LIST_INIT(&toremove_connections[i]);
+ idle_conns[i].cleanup_task->process = srv_cleanup_toremove_connections;
+ idle_conns[i].cleanup_task->context = NULL;
+ MT_LIST_INIT(&idle_conns[i].toremove_conns);
}
}
DECLARE_POOL(pool_head_sockaddr, "sockaddr", sizeof(struct sockaddr_storage));
DECLARE_POOL(pool_head_authority, "authority", PP2_AUTHORITY_MAX);
+struct idle_conns idle_conns[MAX_THREADS] = { };
struct xprt_ops *registered_xprt[XPRT_ENTRIES] = { NULL, };
/* List head of all known muxes for PROTO */
int ret = 0;
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (tl->context == NULL) {
/* The connection has been taken over by another thread,
* we're no longer responsible for it, so just free the
* tasklet, and do nothing.
*/
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
tasklet_free(tl);
return NULL;
if (conn_in_list)
MT_LIST_DEL(&conn->list);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (!(fconn->wait_event.events & SUB_RETRY_SEND))
ret = fcgi_send(fconn);
/* We're about to destroy the connection, so make sure nobody attempts
* to steal it from us.
*/
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (fconn && fconn->conn->flags & CO_FL_LIST_MASK)
MT_LIST_DEL(&fconn->conn->list);
if (!t->context)
fconn = NULL;
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
task_destroy(t);
int ret = 0;
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (tl->context == NULL) {
/* The connection has been taken over by another thread,
* we're no longer responsible for it, so just free the
* tasklet, and do nothing.
*/
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
tasklet_free(tl);
return NULL;
}
if (conn_in_list)
MT_LIST_DEL(&conn->list);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (!(h1c->wait_event.events & SUB_RETRY_SEND))
ret = h1_send(h1c);
/* We're about to destroy the connection, so make sure nobody attempts
* to steal it from us.
*/
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (h1c && h1c->conn->flags & CO_FL_LIST_MASK)
MT_LIST_DEL(&h1c->conn->list);
if (!t->context)
h1c = NULL;
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
task_destroy(t);
int ret = 0;
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (t->context == NULL) {
/* The connection has been taken over by another thread,
* we're no longer responsible for it, so just free the
* tasklet, and do nothing.
*/
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
tasklet_free(tl);
goto leave;
}
if (conn_in_list)
MT_LIST_DEL(&conn->list);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (!(h2c->wait_event.events & SUB_RETRY_SEND))
ret = h2_send(h2c);
}
/* connections in error must be removed from the idle lists */
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
MT_LIST_DEL((struct mt_list *)&conn->list);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
}
else if (h2c->st0 == H2_CS_ERROR) {
/* connections in error must be removed from the idle lists */
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
MT_LIST_DEL((struct mt_list *)&conn->list);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
}
if (!b_data(&h2c->dbuf))
/* We're about to destroy the connection, so make sure nobody attempts
* to steal it from us.
*/
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
if (h2c && h2c->conn->flags & CO_FL_LIST_MASK)
MT_LIST_DEL(&h2c->conn->list);
if (!t->context)
h2c = NULL;
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
task_destroy(t);
}
/* in any case this connection must not be considered idle anymore */
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
MT_LIST_DEL((struct mt_list *)&h2c->conn->list);
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[tid]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[tid].toremove_lock);
/* either we can release everything now or it will be done later once
* the last stream closes.
__decl_thread(HA_SPINLOCK_T idle_conn_srv_lock);
struct eb_root idle_conn_srv = EB_ROOT;
struct task *idle_conn_task = NULL;
-struct task *idle_conn_cleanup[MAX_THREADS] = { NULL };
-struct mt_list toremove_connections[MAX_THREADS];
-__decl_thread(HA_SPINLOCK_T toremove_lock[MAX_THREADS]);
/* The server names dictionary */
struct dict server_name_dict = {
{
struct connection *conn;
- while ((conn = MT_LIST_POP(&toremove_connections[tid],
+ while ((conn = MT_LIST_POP(&idle_conns[tid].toremove_conns,
struct connection *, list)) != NULL) {
conn->mux->destroy(conn->ctx);
}
HA_SPIN_LOCK(OTHER_LOCK, &idle_conn_srv_lock);
for (i = 0; i < global.nbthread; i++) {
did_remove = 0;
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[i]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[i].toremove_lock);
for (j = 0; j < srv->curr_idle_conns; j++) {
conn = MT_LIST_POP(&srv->idle_conns[i], struct connection *, list);
if (!conn)
if (!conn)
break;
did_remove = 1;
- MT_LIST_ADDQ(&toremove_connections[i], (struct mt_list *)&conn->list);
+ MT_LIST_ADDQ(&idle_conns[i].toremove_conns, (struct mt_list *)&conn->list);
}
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[i]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[i].toremove_lock);
if (did_remove)
- task_wakeup(idle_conn_cleanup[i], TASK_WOKEN_OTHER);
+ task_wakeup(idle_conns[i].cleanup_task, TASK_WOKEN_OTHER);
}
HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conn_srv_lock);
}
max_conn = (exceed_conns * srv->curr_idle_thr[i]) /
curr_idle + 1;
- HA_SPIN_LOCK(OTHER_LOCK, &toremove_lock[i]);
+ HA_SPIN_LOCK(OTHER_LOCK, &idle_conns[i].toremove_lock);
for (j = 0; j < max_conn; j++) {
struct connection *conn = MT_LIST_POP(&srv->idle_conns[i], struct connection *, list);
if (!conn)
if (!conn)
break;
did_remove = 1;
- MT_LIST_ADDQ(&toremove_connections[i], (struct mt_list *)&conn->list);
+ MT_LIST_ADDQ(&idle_conns[i].toremove_conns, (struct mt_list *)&conn->list);
}
- HA_SPIN_UNLOCK(OTHER_LOCK, &toremove_lock[i]);
+ HA_SPIN_UNLOCK(OTHER_LOCK, &idle_conns[i].toremove_lock);
if (did_remove && max_conn < srv->curr_idle_thr[i])
srv_is_empty = 0;
if (did_remove)
- task_wakeup(idle_conn_cleanup[i], TASK_WOKEN_OTHER);
+ task_wakeup(idle_conns[i].cleanup_task, TASK_WOKEN_OTHER);
}
remove:
eb32_delete(&srv->idle_node);