Willy Tarreau | b1ec8c4 | 2015-04-03 13:53:24 +0200 | [diff] [blame] | 1 | /* |
| 2 | * include/proto/session.h |
| 3 | * This file defines everything related to sessions. |
| 4 | * |
| 5 | * Copyright (C) 2000-2015 Willy Tarreau - w@1wt.eu |
| 6 | * |
| 7 | * This library is free software; you can redistribute it and/or |
| 8 | * modify it under the terms of the GNU Lesser General Public |
| 9 | * License as published by the Free Software Foundation, version 2.1 |
| 10 | * exclusively. |
| 11 | * |
| 12 | * This library is distributed in the hope that it will be useful, |
| 13 | * but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 14 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
| 15 | * Lesser General Public License for more details. |
| 16 | * |
| 17 | * You should have received a copy of the GNU Lesser General Public |
| 18 | * License along with this library; if not, write to the Free Software |
| 19 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
| 20 | */ |
| 21 | |
| 22 | #ifndef _PROTO_SESSION_H |
| 23 | #define _PROTO_SESSION_H |
| 24 | |
| 25 | #include <common/config.h> |
| 26 | #include <common/buffer.h> |
| 27 | #include <common/debug.h> |
| 28 | #include <common/memory.h> |
| 29 | |
| 30 | #include <types/global.h> |
| 31 | #include <types/session.h> |
| 32 | |
Olivier Houchard | a4d4fdf | 2018-12-14 19:27:06 +0100 | [diff] [blame] | 33 | #include <proto/obj_type.h> |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 34 | #include <proto/stick_table.h> |
Olivier Houchard | a4d4fdf | 2018-12-14 19:27:06 +0100 | [diff] [blame] | 35 | #include <proto/server.h> |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 36 | |
Willy Tarreau | bafbe01 | 2017-11-24 17:34:44 +0100 | [diff] [blame] | 37 | extern struct pool_head *pool_head_session; |
Willy Tarreau | c38f71c | 2015-04-05 00:38:48 +0200 | [diff] [blame] | 38 | struct session *session_new(struct proxy *fe, struct listener *li, enum obj_type *origin); |
Willy Tarreau | 11c3624 | 2015-04-04 15:54:03 +0200 | [diff] [blame] | 39 | void session_free(struct session *sess); |
Willy Tarreau | 9903f0e | 2015-04-04 18:50:31 +0200 | [diff] [blame] | 40 | int session_accept_fd(struct listener *l, int cfd, struct sockaddr_storage *addr); |
Willy Tarreau | b1ec8c4 | 2015-04-03 13:53:24 +0200 | [diff] [blame] | 41 | |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 42 | /* Remove the refcount from the session to the tracked counters, and clear the |
| 43 | * pointer to ensure this is only performed once. The caller is responsible for |
| 44 | * ensuring that the pointer is valid first. |
| 45 | */ |
| 46 | static inline void session_store_counters(struct session *sess) |
| 47 | { |
| 48 | void *ptr; |
| 49 | int i; |
Emeric Brun | 819fc6f | 2017-06-13 19:37:32 +0200 | [diff] [blame] | 50 | struct stksess *ts; |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 51 | |
| 52 | for (i = 0; i < MAX_SESS_STKCTR; i++) { |
| 53 | struct stkctr *stkctr = &sess->stkctr[i]; |
| 54 | |
Emeric Brun | 819fc6f | 2017-06-13 19:37:32 +0200 | [diff] [blame] | 55 | ts = stkctr_entry(stkctr); |
| 56 | if (!ts) |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 57 | continue; |
| 58 | |
Emeric Brun | 819fc6f | 2017-06-13 19:37:32 +0200 | [diff] [blame] | 59 | ptr = stktable_data_ptr(stkctr->table, ts, STKTABLE_DT_CONN_CUR); |
| 60 | if (ptr) { |
Christopher Faulet | 2a944ee | 2017-11-07 10:42:54 +0100 | [diff] [blame] | 61 | HA_RWLOCK_WRLOCK(STK_SESS_LOCK, &ts->lock); |
Emeric Brun | 819fc6f | 2017-06-13 19:37:32 +0200 | [diff] [blame] | 62 | |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 63 | stktable_data_cast(ptr, conn_cur)--; |
Emeric Brun | 819fc6f | 2017-06-13 19:37:32 +0200 | [diff] [blame] | 64 | |
Christopher Faulet | 2a944ee | 2017-11-07 10:42:54 +0100 | [diff] [blame] | 65 | HA_RWLOCK_WRUNLOCK(STK_SESS_LOCK, &ts->lock); |
Emeric Brun | 0fed0b0 | 2017-11-29 16:15:07 +0100 | [diff] [blame] | 66 | |
| 67 | /* If data was modified, we need to touch to re-schedule sync */ |
| 68 | stktable_touch_local(stkctr->table, ts, 0); |
Emeric Brun | 819fc6f | 2017-06-13 19:37:32 +0200 | [diff] [blame] | 69 | } |
| 70 | |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 71 | stkctr_set_entry(stkctr, NULL); |
Emeric Brun | 819fc6f | 2017-06-13 19:37:32 +0200 | [diff] [blame] | 72 | stksess_kill_if_expired(stkctr->table, ts, 1); |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 73 | } |
| 74 | } |
| 75 | |
Olivier Houchard | 00cf70f | 2018-11-30 17:24:55 +0100 | [diff] [blame] | 76 | static inline void session_add_conn(struct session *sess, struct connection *conn, void *target) |
| 77 | { |
| 78 | int avail = -1; |
| 79 | int i; |
| 80 | |
Olivier Houchard | a4d4fdf | 2018-12-14 19:27:06 +0100 | [diff] [blame] | 81 | sess->resp_conns++; |
Olivier Houchard | 00cf70f | 2018-11-30 17:24:55 +0100 | [diff] [blame] | 82 | for (i = 0; i < MAX_SRV_LIST; i++) { |
| 83 | if (sess->srv_list[i].target == target) { |
| 84 | avail = i; |
| 85 | break; |
| 86 | } |
| 87 | if (LIST_ISEMPTY(&sess->srv_list[i].list) && avail == -1) |
| 88 | avail = i; |
| 89 | } |
| 90 | if (avail == -1) { |
| 91 | struct connection *conn, *conn_back; |
| 92 | int count = 0; |
| 93 | /* We have no slot free, let's free the one with the fewer connections */ |
| 94 | for (i = 0; i < MAX_SRV_LIST; i++) { |
| 95 | int count_list = 0; |
| 96 | list_for_each_entry(conn, &sess->srv_list[i].list, session_list) |
| 97 | count_list++; |
| 98 | if (count == 0 || count_list < count) { |
| 99 | count = count_list; |
| 100 | avail = i; |
| 101 | } |
| 102 | } |
| 103 | /* Now unown all the connections */ |
| 104 | list_for_each_entry_safe(conn, conn_back, &sess->srv_list[avail].list, session_list) { |
Olivier Houchard | a4d4fdf | 2018-12-14 19:27:06 +0100 | [diff] [blame] | 105 | sess->resp_conns--; |
Olivier Houchard | 00cf70f | 2018-11-30 17:24:55 +0100 | [diff] [blame] | 106 | conn->owner = NULL; |
| 107 | LIST_DEL(&conn->session_list); |
| 108 | LIST_INIT(&conn->session_list); |
| 109 | if (conn->mux) |
| 110 | conn->mux->destroy(conn); |
| 111 | } |
| 112 | |
| 113 | } |
| 114 | sess->srv_list[avail].target = target; |
| 115 | LIST_ADDQ(&sess->srv_list[avail].list, &conn->session_list); |
| 116 | } |
| 117 | |
Olivier Houchard | a4d4fdf | 2018-12-14 19:27:06 +0100 | [diff] [blame] | 118 | /* 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 */ |
| 119 | static inline int session_check_idle_conn(struct session *sess, struct connection *conn) |
| 120 | { |
| 121 | if (sess->resp_conns > sess->fe->max_out_conns) { |
| 122 | /* We can't keep the connection, let's try to add it to the server idle list */ |
| 123 | LIST_DEL(&conn->session_list); |
| 124 | LIST_INIT(&conn->session_list); |
| 125 | conn->owner = NULL; |
| 126 | sess->resp_conns--; |
| 127 | if (!srv_add_to_idle_list(objt_server(conn->target), conn)) { |
| 128 | /* The server doesn't want it, let's kill the connection right away */ |
| 129 | conn->mux->destroy(conn); |
| 130 | return -1; |
| 131 | } |
| 132 | return 1; |
| 133 | } |
| 134 | return 0; |
| 135 | } |
Willy Tarreau | bb2ef12 | 2015-04-04 16:31:16 +0200 | [diff] [blame] | 136 | |
Willy Tarreau | b1ec8c4 | 2015-04-03 13:53:24 +0200 | [diff] [blame] | 137 | #endif /* _PROTO_SESSION_H */ |
| 138 | |
| 139 | /* |
| 140 | * Local variables: |
| 141 | * c-indent-level: 8 |
| 142 | * c-basic-offset: 8 |
| 143 | * End: |
| 144 | */ |