mirror of
				https://github.com/espressif/esp-mqtt.git
				synced 2025-10-31 07:02:30 +01:00 
			
		
		
		
	feat: Introduces outbox limit
A memory limit for the outbox can be configured. User will not be able to publish or enqueue if the new message goes beyond the configured limit.
This commit is contained in:
		| @@ -1,4 +1,5 @@ | ||||
| #include "mqtt_outbox.h" | ||||
| #include <stdint.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
| #include "mqtt_config.h" | ||||
| @@ -22,12 +23,19 @@ typedef struct outbox_item { | ||||
|  | ||||
| STAILQ_HEAD(outbox_list_t, outbox_item); | ||||
|  | ||||
| struct outbox_t { | ||||
|     uint64_t size; | ||||
|     struct outbox_list_t *list; | ||||
| }; | ||||
|  | ||||
| outbox_handle_t outbox_init(void) | ||||
| { | ||||
|     outbox_handle_t outbox = calloc(1, sizeof(struct outbox_list_t)); | ||||
|     outbox_handle_t outbox = calloc(1, sizeof(struct outbox_t)); | ||||
|     ESP_MEM_CHECK(TAG, outbox, return NULL); | ||||
|     STAILQ_INIT(outbox); | ||||
|     outbox->list = calloc(1, sizeof(struct outbox_list_t)); | ||||
|     ESP_MEM_CHECK(TAG, outbox->list, return NULL); //TODO: Free outbox on failure | ||||
|     outbox->size = 0; | ||||
|     STAILQ_INIT(outbox->list); | ||||
|     return outbox; | ||||
| } | ||||
|  | ||||
| @@ -50,7 +58,8 @@ outbox_item_handle_t outbox_enqueue(outbox_handle_t outbox, outbox_message_handl | ||||
|     if (message->remaining_data) { | ||||
|         memcpy(item->buffer + message->len, message->remaining_data, message->remaining_len); | ||||
|     } | ||||
|     STAILQ_INSERT_TAIL(outbox, item, next); | ||||
|     STAILQ_INSERT_TAIL(outbox->list, item, next); | ||||
|     outbox->size += item->len; | ||||
|     ESP_LOGD(TAG, "ENQUEUE msgid=%d, msg_type=%d, len=%d, size=%d", message->msg_id, message->msg_type, message->len + message->remaining_len, outbox_get_size(outbox)); | ||||
|     return item; | ||||
| } | ||||
| @@ -58,7 +67,7 @@ outbox_item_handle_t outbox_enqueue(outbox_handle_t outbox, outbox_message_handl | ||||
| outbox_item_handle_t outbox_get(outbox_handle_t outbox, int msg_id) | ||||
| { | ||||
|     outbox_item_handle_t item; | ||||
|     STAILQ_FOREACH(item, outbox, next) { | ||||
|     STAILQ_FOREACH(item, outbox->list, next) { | ||||
|         if (item->msg_id == msg_id) { | ||||
|             return item; | ||||
|         } | ||||
| @@ -69,7 +78,7 @@ outbox_item_handle_t outbox_get(outbox_handle_t outbox, int msg_id) | ||||
| outbox_item_handle_t outbox_dequeue(outbox_handle_t outbox, pending_state_t pending, outbox_tick_t *tick) | ||||
| { | ||||
|     outbox_item_handle_t item; | ||||
|     STAILQ_FOREACH(item, outbox, next) { | ||||
|     STAILQ_FOREACH(item, outbox->list, next) { | ||||
|         if (item->pending == pending) { | ||||
|             if (tick) { | ||||
|                 *tick = item->tick; | ||||
| @@ -83,9 +92,10 @@ outbox_item_handle_t outbox_dequeue(outbox_handle_t outbox, pending_state_t pend | ||||
| esp_err_t outbox_delete_item(outbox_handle_t outbox, outbox_item_handle_t item_to_delete) | ||||
| { | ||||
|     outbox_item_handle_t item; | ||||
|     STAILQ_FOREACH(item, outbox, next) { | ||||
|     STAILQ_FOREACH(item, outbox->list, next) { | ||||
|         if (item == item_to_delete) { | ||||
|             STAILQ_REMOVE(outbox, item, outbox_item, next); | ||||
|             STAILQ_REMOVE(outbox->list, item, outbox_item, next); | ||||
|             outbox->size -= item->len; | ||||
|             free(item->buffer); | ||||
|             free(item); | ||||
|             return ESP_OK; | ||||
| @@ -109,9 +119,10 @@ uint8_t *outbox_item_get_data(outbox_item_handle_t item,  size_t *len, uint16_t | ||||
| esp_err_t outbox_delete(outbox_handle_t outbox, int msg_id, int msg_type) | ||||
| { | ||||
|     outbox_item_handle_t item, tmp; | ||||
|     STAILQ_FOREACH_SAFE(item, outbox, next, tmp) { | ||||
|     STAILQ_FOREACH_SAFE(item, outbox->list, next, tmp) { | ||||
|         if (item->msg_id == msg_id && (0xFF & (item->msg_type)) == msg_type) { | ||||
|             STAILQ_REMOVE(outbox, item, outbox_item, next); | ||||
|             STAILQ_REMOVE(outbox->list, item, outbox_item, next); | ||||
|             outbox->size -= item->len; | ||||
|             free(item->buffer); | ||||
|             free(item); | ||||
|             ESP_LOGD(TAG, "DELETED msgid=%d, msg_type=%d, remain size=%d", msg_id, msg_type, outbox_get_size(outbox)); | ||||
| @@ -154,10 +165,11 @@ int outbox_delete_single_expired(outbox_handle_t outbox, outbox_tick_t current_t | ||||
| { | ||||
|     int msg_id = -1; | ||||
|     outbox_item_handle_t item; | ||||
|     STAILQ_FOREACH(item, outbox, next) { | ||||
|     STAILQ_FOREACH(item, outbox->list, next) { | ||||
|         if (current_tick - item->tick > timeout) { | ||||
|             STAILQ_REMOVE(outbox, item, outbox_item, next); | ||||
|             STAILQ_REMOVE(outbox->list, item, outbox_item, next); | ||||
|             free(item->buffer); | ||||
|             outbox->size -= item->len; | ||||
|             msg_id = item->msg_id; | ||||
|             free(item); | ||||
|             return msg_id; | ||||
| @@ -171,10 +183,11 @@ int outbox_delete_expired(outbox_handle_t outbox, outbox_tick_t current_tick, ou | ||||
| { | ||||
|     int deleted_items = 0; | ||||
|     outbox_item_handle_t item, tmp; | ||||
|     STAILQ_FOREACH_SAFE(item, outbox, next, tmp) { | ||||
|     STAILQ_FOREACH_SAFE(item, outbox->list, next, tmp) { | ||||
|         if (current_tick - item->tick > timeout) { | ||||
|             STAILQ_REMOVE(outbox, item, outbox_item, next); | ||||
|             STAILQ_REMOVE(outbox->list, item, outbox_item, next); | ||||
|             free(item->buffer); | ||||
|             outbox->size -= item->len; | ||||
|             free(item); | ||||
|             deleted_items ++; | ||||
|         } | ||||
| @@ -183,23 +196,17 @@ int outbox_delete_expired(outbox_handle_t outbox, outbox_tick_t current_tick, ou | ||||
|     return deleted_items; | ||||
| } | ||||
|  | ||||
| int outbox_get_size(outbox_handle_t outbox) | ||||
| uint64_t outbox_get_size(outbox_handle_t outbox) | ||||
| { | ||||
|     int siz = 0; | ||||
|     outbox_item_handle_t item; | ||||
|     STAILQ_FOREACH(item, outbox, next) { | ||||
|         // Suppressing "use after free" warning as this could happen only if queue is in inconsistent state | ||||
|         // which never happens if STAILQ interface used | ||||
|         siz += item->len; // NOLINT(clang-analyzer-unix.Malloc) | ||||
|     } | ||||
|     return siz; | ||||
|     return outbox->size; | ||||
| } | ||||
|  | ||||
| void outbox_delete_all_items(outbox_handle_t outbox) | ||||
| { | ||||
|     outbox_item_handle_t item, tmp; | ||||
|     STAILQ_FOREACH_SAFE(item, outbox, next, tmp) { | ||||
|         STAILQ_REMOVE(outbox, item, outbox_item, next); | ||||
|     STAILQ_FOREACH_SAFE(item, outbox->list, next, tmp) { | ||||
|         STAILQ_REMOVE(outbox->list, item, outbox_item, next); | ||||
|         outbox->size -= item->len; | ||||
|         free(item->buffer); | ||||
|         free(item); | ||||
|     } | ||||
| @@ -207,6 +214,7 @@ void outbox_delete_all_items(outbox_handle_t outbox) | ||||
| void outbox_destroy(outbox_handle_t outbox) | ||||
| { | ||||
|     outbox_delete_all_items(outbox); | ||||
|     free(outbox->list); | ||||
|     free(outbox); | ||||
| } | ||||
|  | ||||
|   | ||||
		Reference in New Issue
	
	Block a user