[Asterisk-code-review] Add data buffer API to store packets. (asterisk[15])

Jenkins2 asteriskteam at digium.com
Thu Mar 29 13:42:31 CDT 2018


Jenkins2 has submitted this change and it was merged. ( https://gerrit.asterisk.org/8604 )

Change subject: Add data buffer API to store packets.
......................................................................

Add data buffer API to store packets.

Adds a data buffer with a configurable size that can store different
kinds of packets (like RTP packets for retransmission). Given a number
it will store a data packet at that position relative to the others.
Given a number it will retrieve the given data packet if it is present.
This is purposely a storage of arbitrary things so it can be used not
just for RTP packets but also Asterisk frames in the future if needed.
The API does not internally use a lock, so it will be up to the user of
the API to properly protect the data buffer.

For more information, refer to the wiki page:
https://wiki.asterisk.org/wiki/display/AST/WebRTC+User+Experience+Improvements

Change-Id: Iff13c5d4795d52356959fe2a57360cd57dfade07
---
A include/asterisk/data_buffer.h
A main/data_buffer.c
2 files changed, 458 insertions(+), 0 deletions(-)

Approvals:
  Sean Bright: Looks good to me, but someone else must approve
  Kevin Harwell: Looks good to me, approved
  Jenkins2: Approved for Submit



diff --git a/include/asterisk/data_buffer.h b/include/asterisk/data_buffer.h
new file mode 100644
index 0000000..dacbaa5
--- /dev/null
+++ b/include/asterisk/data_buffer.h
@@ -0,0 +1,144 @@
+/*
+ * Asterisk -- An open source telephony toolkit.
+ *
+ * Copyright (C) 2018, Digium, Inc.
+ *
+ * Joshua Colp <jcolp at digium.com>
+ * Ben Ford <bford at digium.com>
+ *
+ * See http://www.asterisk.org for more information about
+ * the Asterisk project. Please do not directly contact
+ * any of the maintainers of this project for assistance;
+ * the project provides a web site, mailing lists and IRC
+ * channels for your use.
+ *
+ * This program is free software, distributed under the terms of
+ * the GNU General Public License Version 2. See the LICENSE file
+ * at the top of the source tree.
+ */
+
+/*!
+ * \file
+ * \brief Data Buffer API
+ *
+ * A data buffer acts as a ring buffer of data. It is given a fixed
+ * number of data packets to store (which may be dynamically changed).
+ * Given a number it will store a data packet at that position relative
+ * to the others. Given a number it will retrieve the given data packet
+ * if it is present. This is purposely a storage of arbitrary things so
+ * that it can be used for multiple things.
+ *
+ * \author Joshua Colp <jcolp at digium.com>
+ * \author Ben Ford <bford at digium.com>
+ */
+
+#ifndef _AST_DATA_BUFFER_H_
+#define _AST_DATA_BUFFER_H_
+
+/*!
+ * \brief A buffer of data payloads.
+ */
+struct ast_data_buffer;
+
+/*!
+ * \brief A callback function to free a data payload in a data buffer
+ *
+ * \param The data payload
+ */
+typedef void (*ast_data_buffer_free_callback)(void *data);
+
+/*!
+ * \brief Allocate a data buffer
+ *
+ * \param free_fn Callback function to free a data payload
+ * \param size The maximum number of data payloads to contain in the data buffer
+ *
+ * \retval non-NULL success
+ * \retval NULL failure
+ *
+ * \note free_fn can be NULL. It is up to the consumer of this API to ensure that memory is
+ * managed appropriately.
+ *
+ * \since 15.4.0
+ */
+struct ast_data_buffer *ast_data_buffer_alloc(ast_data_buffer_free_callback free_fn, size_t size);
+
+/*!
+ * \brief Resize a data buffer
+ *
+ * \param buffer The data buffer
+ * \param size The new maximum size of the data buffer
+ *
+ * \note If the data buffer is shrunk any old data payloads will be freed using the configured callback.
+ * The data buffer is flexible and can be used for multiple purposes. Therefore it is up to the
+ * caller of the function to know whether or not a buffer should have its size changed. Increasing
+ * the size of the buffer may make sense in some scenarios, but shrinking should always be handled
+ * with caution since data can be lost.
+ *
+ * \since 15.4.0
+ */
+void ast_data_buffer_resize(struct ast_data_buffer *buffer, size_t size);
+
+/*!
+ * \brief Place a data payload at a position in the data buffer
+ *
+ * \param buffer The data buffer
+ * \param pos The position of the data payload
+ * \param payload The data payload
+ *
+ * \retval 0 success
+ * \retval -1 failure
+ *
+ * \note It is up to the consumer of this API to ensure proper memory management of data payloads
+ *
+ * \since 15.4.0
+ */
+int ast_data_buffer_put(struct ast_data_buffer *buffer, size_t pos, void *payload);
+
+/*!
+ * \brief Retrieve a data payload from the data buffer
+ *
+ * \param buffer The data buffer
+ * \param pos The position of the data payload
+ *
+ * \retval non-NULL success
+ * \retval NULL failure
+ *
+ * \note This does not remove the data payload from the data buffer. It will be removed when it is displaced.
+ *
+ * \since 15.4.0
+ */
+void *ast_data_buffer_get(const struct ast_data_buffer *buffer, size_t pos);
+
+/*!
+ * \brief Free a data buffer (and all held data payloads)
+ *
+ * \param buffer The data buffer
+ *
+ * \since 15.4.0
+ */
+void ast_data_buffer_free(struct ast_data_buffer *buffer);
+
+/*!
+ * \brief Return the number of payloads in a data buffer
+ *
+ * \param buffer The data buffer
+ *
+ * \retval the number of data payloads
+ *
+ * \since 15.4.0
+ */
+size_t ast_data_buffer_count(const struct ast_data_buffer *buffer);
+
+/*!
+ * \brief Return the maximum number of payloads a data buffer can hold
+ *
+ * \param buffer The data buffer
+ *
+ * \retval the maximum number of data payloads
+ *
+ * \since 15.4.0
+ */
+size_t ast_data_buffer_max(const struct ast_data_buffer *buffer);
+
+#endif /* _AST_DATA_BUFFER_H */
diff --git a/main/data_buffer.c b/main/data_buffer.c
new file mode 100644
index 0000000..ccbffd2
--- /dev/null
+++ b/main/data_buffer.c
@@ -0,0 +1,314 @@
+/*
+ * Asterisk -- An open source telephony toolkit.
+ *
+ * Copyright (C) 2018, Digium, Inc.
+ *
+ * Joshua Colp <jcolp at digium.com>
+ * Ben Ford <bford at digium.com>
+ *
+ * See http://www.asterisk.org for more information about
+ * the Asterisk project. Please do not directly contact
+ * any of the maintainers of this project for assistance;
+ * the project provides a web site, mailing lists and IRC
+ * channels for your use.
+ *
+ * This program is free software, distributed under the terms of
+ * the GNU General Public License Version 2. See the LICENSE file
+ * at the top of the source tree.
+ */
+
+/*! \file
+ *
+ * \brief Data Buffer API
+ *
+ * \author Joshua Colp <jcolp at digium.com>
+ * \author Ben Ford <bford at digium.com>
+ */
+
+/*** MODULEINFO
+	<support_level>core</support_level>
+ ***/
+
+#include "asterisk.h"
+
+#include "asterisk/logger.h"
+#include "asterisk/strings.h"
+#include "asterisk/data_buffer.h"
+#include "asterisk/linkedlists.h"
+
+/*!
+ * \brief The number of payloads to increment the cache by
+ */
+#define CACHED_PAYLOAD_MAX 5
+
+/*!
+ * \brief Payload entry placed inside of the data buffer list
+ */
+struct data_buffer_payload_entry {
+	/*! \brief The payload for this position */
+	void *payload;
+	/*! \brief The provided position for this */
+	size_t pos;
+	/*! \brief Linked list information */
+	AST_LIST_ENTRY(data_buffer_payload_entry) list;
+};
+
+/*!
+ * \brief Data buffer containing fixed number of data payloads
+ */
+struct ast_data_buffer {
+	/*! \brief Callback function to free a data payload */
+	ast_data_buffer_free_callback free_fn;
+	/*! \brief A linked list of data payloads */
+	AST_LIST_HEAD_NOLOCK(, data_buffer_payload_entry) payloads;
+	/*! \brief A linked list of unused cached data payloads */
+	AST_LIST_HEAD_NOLOCK(, data_buffer_payload_entry) cached_payloads;
+	/*! \brief The current number of data payloads in the buffer */
+	size_t count;
+	/*! \brief Maximum number of data payloads in the buffer */
+	size_t max;
+	/*! \brief The current number of data payloads in the cache */
+	size_t cache_count;
+};
+
+static void free_fn_do_nothing(void *data)
+{
+	return;
+}
+
+/*!
+ * \brief Helper function to allocate a data payload
+ */
+static struct data_buffer_payload_entry *data_buffer_payload_alloc(void *payload, size_t pos)
+{
+	struct data_buffer_payload_entry *data_payload;
+
+	data_payload = ast_calloc(1, sizeof(*data_payload));
+	if (!data_payload) {
+		return NULL;
+	}
+
+	data_payload->payload = payload;
+	data_payload->pos = pos;
+
+	return data_payload;
+}
+
+/*!
+ * \brief Helper function that sets the cache to its maximum number of payloads
+ */
+static void ast_data_buffer_cache_adjust(struct ast_data_buffer *buffer)
+{
+	int buffer_space;
+
+	ast_assert(buffer != NULL);
+
+	buffer_space = buffer->max - buffer->count;
+
+	if (buffer->cache_count == buffer_space) {
+		return;
+	}
+
+	if (buffer->cache_count < buffer_space) {
+		/* Add payloads to the cache, if able */
+		while (buffer->cache_count < CACHED_PAYLOAD_MAX && buffer->cache_count < buffer_space) {
+			struct data_buffer_payload_entry *buffer_payload;
+
+			buffer_payload = data_buffer_payload_alloc(NULL, -1);
+			if (buffer_payload) {
+				AST_LIST_INSERT_TAIL(&buffer->cached_payloads, buffer_payload, list);
+				buffer->cache_count++;
+				continue;
+			}
+
+			ast_log(LOG_ERROR, "Failed to allocate memory to the cache.");
+			break;
+		}
+	} else if (buffer->cache_count > buffer_space) {
+		/* Remove payloads from the cache */
+		while (buffer->cache_count > buffer_space) {
+			struct data_buffer_payload_entry *buffer_payload;
+
+			buffer_payload = AST_LIST_REMOVE_HEAD(&buffer->cached_payloads, list);
+			if (buffer_payload) {
+				ast_free(buffer_payload);
+				buffer->cache_count--;
+				continue;
+			}
+
+			ast_log(LOG_ERROR, "Failed to remove memory from the cache.");
+			break;
+		}
+	}
+}
+
+struct ast_data_buffer *ast_data_buffer_alloc(ast_data_buffer_free_callback free_fn, size_t size)
+{
+	struct ast_data_buffer *buffer;
+
+	ast_assert(size != 0);
+
+	buffer = ast_calloc(1, sizeof(*buffer));
+	if (!buffer) {
+		return NULL;
+	}
+
+	AST_LIST_HEAD_INIT_NOLOCK(&buffer->payloads);
+	AST_LIST_HEAD_INIT_NOLOCK(&buffer->cached_payloads);
+
+	/* If free_fn is NULL, just use free_fn_do_nothing as a default */
+	buffer->free_fn = free_fn ? free_fn : free_fn_do_nothing;
+	buffer->max = size;
+
+	ast_data_buffer_cache_adjust(buffer);
+
+	return buffer;
+}
+
+void ast_data_buffer_resize(struct ast_data_buffer *buffer, size_t size)
+{
+	struct data_buffer_payload_entry *existing_payload;
+
+	ast_assert(buffer != NULL);
+
+	/* The buffer must have at least a size of 1 */
+	ast_assert(size > 0);
+
+	if (buffer->max == size) {
+		return;
+	}
+
+	/* If the size is decreasing, some payloads will need to be freed */
+	if (buffer->max > size) {
+		int remove = buffer->max - size;
+
+		AST_LIST_TRAVERSE_SAFE_BEGIN(&buffer->payloads, existing_payload, list) {
+			if (remove) {
+				AST_LIST_REMOVE_HEAD(&buffer->payloads, list);
+				buffer->free_fn(existing_payload->payload);
+				ast_free(existing_payload);
+				buffer->count--;
+				remove--;
+				continue;
+			}
+			break;
+		}
+		AST_LIST_TRAVERSE_SAFE_END;
+	}
+
+	buffer->max = size;
+	ast_data_buffer_cache_adjust(buffer);
+}
+
+int ast_data_buffer_put(struct ast_data_buffer *buffer, size_t pos, void *payload)
+{
+	struct data_buffer_payload_entry *buffer_payload = NULL;
+	struct data_buffer_payload_entry *existing_payload;
+	int inserted = 0;
+
+	ast_assert(buffer != NULL);
+	ast_assert(payload != NULL);
+
+	/* If the data buffer has reached its maximum size then the head goes away and
+	 * we will reuse its buffer payload
+	 */
+	if (buffer->count == buffer->max) {
+		buffer_payload = AST_LIST_REMOVE_HEAD(&buffer->payloads, list);
+		buffer->free_fn(buffer_payload->payload);
+		buffer->count--;
+
+		/* Update this buffer payload with its new information */
+		buffer_payload->payload = payload;
+		buffer_payload->pos = pos;
+	}
+	if (!buffer_payload) {
+		if (!buffer->cache_count) {
+			ast_data_buffer_cache_adjust(buffer);
+		}
+		buffer_payload = AST_LIST_REMOVE_HEAD(&buffer->cached_payloads, list);
+		buffer->cache_count--;
+
+		/* Update the payload from the cache with its new information */
+		buffer_payload->payload = payload;
+		buffer_payload->pos = pos;
+	}
+	if (!buffer_payload) {
+		return -1;
+	}
+
+	/* Given the position find its ideal spot within the buffer */
+	AST_LIST_TRAVERSE_SAFE_BEGIN(&buffer->payloads, existing_payload, list) {
+		/* If it's already in the buffer, drop it */
+		if (existing_payload->pos == pos) {
+			ast_debug(3, "Packet with position %zu is already in buffer. Not inserting.\n", pos);
+			inserted = -1;
+			break;
+		}
+
+		if (existing_payload->pos > pos) {
+			AST_LIST_INSERT_BEFORE_CURRENT(buffer_payload, list);
+			inserted = 1;
+			break;
+		}
+	}
+	AST_LIST_TRAVERSE_SAFE_END;
+
+	if (inserted == -1) {
+		return 0;
+	}
+
+	if (!inserted) {
+		AST_LIST_INSERT_TAIL(&buffer->payloads, buffer_payload, list);
+	}
+
+	buffer->count++;
+
+	return 0;
+}
+
+void *ast_data_buffer_get(const struct ast_data_buffer *buffer, size_t pos)
+{
+	struct data_buffer_payload_entry *buffer_payload;
+
+	ast_assert(buffer != NULL);
+
+	AST_LIST_TRAVERSE(&buffer->payloads, buffer_payload, list) {
+		if (buffer_payload->pos == pos) {
+			return buffer_payload->payload;
+		}
+	}
+
+	return NULL;
+}
+
+void ast_data_buffer_free(struct ast_data_buffer *buffer)
+{
+	struct data_buffer_payload_entry *buffer_payload;
+
+	ast_assert(buffer != NULL);
+
+	while ((buffer_payload = AST_LIST_REMOVE_HEAD(&buffer->payloads, list))) {
+		buffer->free_fn(buffer_payload->payload);
+		ast_free(buffer_payload);
+	}
+
+	while ((buffer_payload = AST_LIST_REMOVE_HEAD(&buffer->cached_payloads, list))) {
+		ast_free(buffer_payload);
+	}
+
+	ast_free(buffer);
+}
+
+size_t ast_data_buffer_count(const struct ast_data_buffer *buffer)
+{
+	ast_assert(buffer != NULL);
+
+	return buffer->count;
+}
+
+size_t ast_data_buffer_max(const struct ast_data_buffer *buffer)
+{
+	ast_assert(buffer != NULL);
+
+	return buffer->max;
+}

-- 
To view, visit https://gerrit.asterisk.org/8604
To unsubscribe, visit https://gerrit.asterisk.org/settings

Gerrit-Project: asterisk
Gerrit-Branch: 15
Gerrit-MessageType: merged
Gerrit-Change-Id: Iff13c5d4795d52356959fe2a57360cd57dfade07
Gerrit-Change-Number: 8604
Gerrit-PatchSet: 11
Gerrit-Owner: Benjamin Keith Ford <bford at digium.com>
Gerrit-Reviewer: Benjamin Keith Ford <bford at digium.com>
Gerrit-Reviewer: George Joseph <gjoseph at digium.com>
Gerrit-Reviewer: Jenkins2
Gerrit-Reviewer: Joshua Colp <jcolp at digium.com>
Gerrit-Reviewer: Kevin Harwell <kharwell at digium.com>
Gerrit-Reviewer: Matthew Fredrickson <creslin at digium.com>
Gerrit-Reviewer: Sean Bright <sean.bright at gmail.com>
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.digium.com/pipermail/asterisk-code-review/attachments/20180329/a522e007/attachment-0001.html>


More information about the asterisk-code-review mailing list