From 4a7f37105659383e154d38d89fbc89cc26e29d9a Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Friedolin=20Gr=C3=B6ger?= Date: Tue, 16 Jul 2024 10:46:50 +0200 Subject: [PATCH] fix(esp_http_server): prevent concurrent access to socket used in async http requests --- components/esp_http_server/src/httpd_main.c | 5 +++++ components/esp_http_server/src/httpd_sess.c | 3 ++- components/esp_http_server/src/httpd_txrx.c | 6 ++++-- 3 files changed, 11 insertions(+), 3 deletions(-) diff --git a/components/esp_http_server/src/httpd_main.c b/components/esp_http_server/src/httpd_main.c index a1006feb368f..fd434e87ae12 100644 --- a/components/esp_http_server/src/httpd_main.c +++ b/components/esp_http_server/src/httpd_main.c @@ -254,6 +254,11 @@ static int httpd_process_session(struct sock_db *session, void *context) return 1; } + // session is busy in an async task, do not process here. + if (session->for_async_req) { + return 1; + } + process_session_context_t *ctx = (process_session_context_t *)context; int fd = session->fd; diff --git a/components/esp_http_server/src/httpd_sess.c b/components/esp_http_server/src/httpd_sess.c index a10c62fb1a19..ac03edf8d159 100644 --- a/components/esp_http_server/src/httpd_sess.c +++ b/components/esp_http_server/src/httpd_sess.c @@ -72,6 +72,7 @@ static int enum_function(struct sock_db *session, void *context) case HTTPD_TASK_INIT: session->fd = -1; session->ctx = NULL; + session->for_async_req = false; break; // Get active session case HTTPD_TASK_GET_ACTIVE: @@ -87,7 +88,7 @@ static int enum_function(struct sock_db *session, void *context) break; // Set descriptor case HTTPD_TASK_SET_DESCRIPTOR: - if (session->fd != -1) { + if (session->fd != -1 && !session->for_async_req) { FD_SET(session->fd, ctx->fdset); if (session->fd > ctx->max_fd) { ctx->max_fd = session->fd; diff --git a/components/esp_http_server/src/httpd_txrx.c b/components/esp_http_server/src/httpd_txrx.c index ad5e8ed97393..fa50378f69e1 100644 --- a/components/esp_http_server/src/httpd_txrx.c +++ b/components/esp_http_server/src/httpd_txrx.c @@ -631,9 +631,11 @@ esp_err_t httpd_req_async_handler_begin(httpd_req_t *r, httpd_req_t **out) } memcpy(async_aux->resp_hdrs, r_aux->resp_hdrs, hd->config.max_resp_headers * sizeof(struct resp_hdr)); + // Prevent the main thread from reading the rest of the request after the handler returns. + r_aux->remaining_len = 0; + // mark socket as "in use" - struct httpd_req_aux *ra = r->aux; - ra->sd->for_async_req = true; + r_aux->sd->for_async_req = true; *out = async;