BUG/MINOR: mux-quic: Fix memleak on QUIC stream buffer for unacknowledged data
Some clients send CONNECTION_CLOSE frame without acknowledging the STREAM data haproxy has sent. In this case, when closing the connection if there were remaining data in QUIC stream buffers, they were not released. Add a <closing> boolean option to qc_stream_desc_free() to force the stream buffer memory releasing upon closing connection. Thank you to Tristan for having reported such a memory leak issue in GH #1801. Must be backported to 2.6.
This commit is contained in:
parent
f53201940b
commit
ea4a5cbbdf
|
@ -12,7 +12,7 @@ struct qc_stream_desc *qc_stream_desc_new(uint64_t id, enum qcs_type, void *ctx,
|
||||||
struct quic_conn *qc);
|
struct quic_conn *qc);
|
||||||
void qc_stream_desc_release(struct qc_stream_desc *stream);
|
void qc_stream_desc_release(struct qc_stream_desc *stream);
|
||||||
int qc_stream_desc_ack(struct qc_stream_desc **stream, size_t offset, size_t len);
|
int qc_stream_desc_ack(struct qc_stream_desc **stream, size_t offset, size_t len);
|
||||||
void qc_stream_desc_free(struct qc_stream_desc *stream);
|
void qc_stream_desc_free(struct qc_stream_desc *stream, int closing);
|
||||||
|
|
||||||
struct buffer *qc_stream_buf_get(struct qc_stream_desc *stream);
|
struct buffer *qc_stream_buf_get(struct qc_stream_desc *stream);
|
||||||
struct buffer *qc_stream_buf_alloc(struct qc_stream_desc *stream,
|
struct buffer *qc_stream_buf_alloc(struct qc_stream_desc *stream,
|
||||||
|
|
|
@ -63,7 +63,7 @@ void qc_stream_desc_release(struct qc_stream_desc *stream)
|
||||||
|
|
||||||
if (LIST_ISEMPTY(&stream->buf_list)) {
|
if (LIST_ISEMPTY(&stream->buf_list)) {
|
||||||
/* if no buffer left we can free the stream. */
|
/* if no buffer left we can free the stream. */
|
||||||
qc_stream_desc_free(stream);
|
qc_stream_desc_free(stream, 0);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
/* A released stream does not use <stream.buf>. */
|
/* A released stream does not use <stream.buf>. */
|
||||||
|
@ -131,7 +131,7 @@ int qc_stream_desc_ack(struct qc_stream_desc **stream, size_t offset, size_t len
|
||||||
|
|
||||||
/* Free stream instance if already released and no buffers left. */
|
/* Free stream instance if already released and no buffers left. */
|
||||||
if (s->release && LIST_ISEMPTY(&s->buf_list)) {
|
if (s->release && LIST_ISEMPTY(&s->buf_list)) {
|
||||||
qc_stream_desc_free(s);
|
qc_stream_desc_free(s, 0);
|
||||||
*stream = NULL;
|
*stream = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -139,10 +139,10 @@ int qc_stream_desc_ack(struct qc_stream_desc **stream, size_t offset, size_t len
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Free the stream descriptor <stream> content. This function should be used
|
/* Free the stream descriptor <stream> content. This function should be used
|
||||||
* when all its data have been acknowledged or on full connection closing. It
|
* when all its data have been acknowledged or on full connection closing if <closing>
|
||||||
* must only be called after the stream is released.
|
* boolean is set to 1. It must only be called after the stream is released.
|
||||||
*/
|
*/
|
||||||
void qc_stream_desc_free(struct qc_stream_desc *stream)
|
void qc_stream_desc_free(struct qc_stream_desc *stream, int closing)
|
||||||
{
|
{
|
||||||
struct qc_stream_buf *buf, *buf_back;
|
struct qc_stream_buf *buf, *buf_back;
|
||||||
struct quic_conn *qc = stream->qc;
|
struct quic_conn *qc = stream->qc;
|
||||||
|
@ -154,7 +154,7 @@ void qc_stream_desc_free(struct qc_stream_desc *stream)
|
||||||
|
|
||||||
/* free remaining stream buffers */
|
/* free remaining stream buffers */
|
||||||
list_for_each_entry_safe(buf, buf_back, &stream->buf_list, list) {
|
list_for_each_entry_safe(buf, buf_back, &stream->buf_list, list) {
|
||||||
if (!(b_data(&buf->buf))) {
|
if (!(b_data(&buf->buf)) || closing) {
|
||||||
b_free(&buf->buf);
|
b_free(&buf->buf);
|
||||||
LIST_DELETE(&buf->list);
|
LIST_DELETE(&buf->list);
|
||||||
pool_free(pool_head_quic_stream_buf, buf);
|
pool_free(pool_head_quic_stream_buf, buf);
|
||||||
|
|
|
@ -4457,7 +4457,7 @@ static void quic_conn_release(struct quic_conn *qc)
|
||||||
* qc_stream_desc_free will liberate the stream instance.
|
* qc_stream_desc_free will liberate the stream instance.
|
||||||
*/
|
*/
|
||||||
BUG_ON(!stream->release);
|
BUG_ON(!stream->release);
|
||||||
qc_stream_desc_free(stream);
|
qc_stream_desc_free(stream, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Purge Rx packet list. */
|
/* Purge Rx packet list. */
|
||||||
|
|
Loading…
Reference in New Issue