MEDIUM: sessions: Keep track of which connections are idle.

Instead of keeping track of the number of connections we're responsible for,
keep track of the number of connections we're responsible for that we are
currently considering idling (ie that we are not using, they may be in use
by other sessions), that way we can actually reuse connections when we have
more connections than the max configured.
This commit is contained in:
Olivier Houchard 2018-12-28 18:50:57 +01:00 committed by Willy Tarreau
parent c685d700fd
commit a2dbeb22fc
6 changed files with 19 additions and 6 deletions

View File

@ -676,7 +676,8 @@ static inline void conn_free(struct connection *conn)
/* Remove ourself from the session's connections list, if any. */
if (!LIST_ISEMPTY(&conn->session_list)) {
struct session *sess = conn->owner;
sess->resp_conns--;
if (conn->flags & CO_FL_SESS_IDLE)
sess->idle_conns--;
session_unown_conn(sess, conn);
}

View File

@ -112,7 +112,6 @@ static inline int session_add_conn(struct session *sess, struct connection *conn
LIST_INIT(&srv_list->conn_list);
LIST_ADDQ(&sess->srv_list, &srv_list->srv_list);
}
sess->resp_conns++;
LIST_ADDQ(&srv_list->conn_list, &conn->session_list);
return 1;
}
@ -120,17 +119,19 @@ static inline int session_add_conn(struct session *sess, struct connection *conn
/* Returns 0 if the session can keep the idle conn, -1 if it was destroyed, or 1 if it was added to the server list */
static inline int session_check_idle_conn(struct session *sess, struct connection *conn)
{
if (sess->resp_conns > sess->fe->max_out_conns) {
if (sess->idle_conns > sess->fe->max_out_conns) {
/* We can't keep the connection, let's try to add it to the server idle list */
session_unown_conn(sess, conn);
conn->owner = NULL;
sess->resp_conns--;
if (!srv_add_to_idle_list(objt_server(conn->target), conn)) {
/* The server doesn't want it, let's kill the connection right away */
conn->mux->destroy(conn);
return -1;
}
return 1;
} else {
conn->flags |= CO_FL_SESS_IDLE;
sess->idle_conns++;
}
return 0;
}

View File

@ -196,6 +196,8 @@ enum {
/* This flag is used to know that a PROXY protocol header was sent by the client */
CO_FL_RCVD_PROXY = 0x20000000,
/* The connection is unused by its owner */
CO_FL_SESS_IDLE = 0x40000000,
/* unused : 0x40000000 */
/* This last flag indicates that the transport layer is used (for instance

View File

@ -55,7 +55,7 @@ struct session {
struct vars vars; /* list of variables for the session scope. */
struct task *task; /* handshake timeout processing */
long t_handshake; /* handshake duration, -1 = not completed */
int resp_conns; /* Number of connections we're currently responsible for */
int idle_conns; /* Number of connections we're currently responsible for that we are not using */
struct list srv_list; /* List of servers and the connections the session is currently responsible for */
};

View File

@ -1240,6 +1240,13 @@ int connect_server(struct stream *s)
LIST_ADDQ(&srv->idle_conns[tid], &srv_conn->list);
if (LIST_ISEMPTY(&srv->idle_orphan_conns[tid]))
task_unlink_wq(srv->idle_task[tid]);
} else if (reuse) {
if (srv_conn->flags & CO_FL_SESS_IDLE) {
struct session *sess = srv_conn->owner;
srv_conn->flags &= ~CO_FL_SESS_IDLE;
sess->idle_conns--;
}
}
/* We're about to use another connection, let the mux know we're
@ -1251,6 +1258,8 @@ int connect_server(struct stream *s)
if (sess) {
if (old_conn && !(old_conn->flags & CO_FL_PRIVATE) &&
old_conn->mux != NULL) {
if (old_conn->flags & CO_FL_SESS_IDLE)
s->sess->idle_conns--;
session_unown_conn(s->sess, old_conn);
old_conn->owner = sess;
if (!session_add_conn(sess, old_conn, s->target)) {

View File

@ -62,7 +62,7 @@ struct session *session_new(struct proxy *fe, struct listener *li, enum obj_type
HA_ATOMIC_ADD(&totalconn, 1);
HA_ATOMIC_ADD(&jobs, 1);
LIST_INIT(&sess->srv_list);
sess->resp_conns = 0;
sess->idle_conns = 0;
}
return sess;
}