2005-11-19 20:07:43 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
* Copyright (C) 2005/2006, Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
*
|
|
|
|
* Version: MPL 1.1
|
|
|
|
*
|
|
|
|
* The contents of this file are subject to the Mozilla Public License Version
|
|
|
|
* 1.1 (the "License"); you may not use this file except in compliance with
|
|
|
|
* the License. You may obtain a copy of the License at
|
|
|
|
* http://www.mozilla.org/MPL/
|
|
|
|
*
|
|
|
|
* Software distributed under the License is distributed on an "AS IS" basis,
|
|
|
|
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
|
|
|
|
* for the specific language governing rights and limitations under the
|
|
|
|
* License.
|
|
|
|
*
|
|
|
|
* The Original Code is FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
|
|
|
*
|
|
|
|
* The Initial Developer of the Original Code is
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
|
|
|
* Anthony Minessale II <anthmct@yahoo.com>
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* switch_buffer.c -- Data Buffering Code
|
|
|
|
*
|
|
|
|
*/
|
2006-04-29 01:00:52 +00:00
|
|
|
#include <switch.h>
|
2006-04-29 06:05:03 +00:00
|
|
|
#include <switch_buffer.h>
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-04-19 19:04:30 +00:00
|
|
|
static uint32_t buffer_id = 0;
|
|
|
|
|
2006-09-08 18:57:24 +00:00
|
|
|
typedef enum {
|
|
|
|
SWITCH_BUFFER_FLAG_DYNAMIC = (1 << 0)
|
|
|
|
} switch_buffer_flag_t;
|
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
struct switch_buffer {
|
2006-09-14 00:15:03 +00:00
|
|
|
switch_byte_t *data;
|
2006-09-21 19:52:56 +00:00
|
|
|
switch_byte_t *head;
|
2006-03-30 23:02:50 +00:00
|
|
|
switch_size_t used;
|
2006-09-22 02:10:27 +00:00
|
|
|
switch_size_t actually_used;
|
2006-03-30 23:02:50 +00:00
|
|
|
switch_size_t datalen;
|
2006-09-08 18:57:24 +00:00
|
|
|
switch_size_t max_len;
|
|
|
|
switch_size_t blocksize;
|
|
|
|
uint32_t flags;
|
2006-04-19 19:04:30 +00:00
|
|
|
uint32_t id;
|
2005-11-19 20:07:43 +00:00
|
|
|
};
|
|
|
|
|
2006-04-29 23:43:28 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_buffer_create(switch_memory_pool_t *pool, switch_buffer_t **buffer, switch_size_t max_len)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-04-29 06:05:03 +00:00
|
|
|
switch_buffer_t *new_buffer;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
2006-04-29 06:05:03 +00:00
|
|
|
if ((new_buffer = switch_core_alloc(pool, sizeof(switch_buffer_t))) != 0
|
2006-02-20 00:23:25 +00:00
|
|
|
&& (new_buffer->data = switch_core_alloc(pool, max_len)) != 0) {
|
2005-11-19 20:07:43 +00:00
|
|
|
new_buffer->datalen = max_len;
|
2006-04-19 19:04:30 +00:00
|
|
|
new_buffer->id = buffer_id++;
|
2006-09-21 19:52:56 +00:00
|
|
|
new_buffer->head = new_buffer->data;
|
2005-11-19 20:07:43 +00:00
|
|
|
*buffer = new_buffer;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
return SWITCH_STATUS_MEMERR;
|
|
|
|
}
|
|
|
|
|
2006-09-08 18:57:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_buffer_create_dynamic(switch_buffer_t **buffer,
|
|
|
|
switch_size_t blocksize,
|
|
|
|
switch_size_t start_len,
|
|
|
|
switch_size_t max_len)
|
|
|
|
{
|
|
|
|
switch_buffer_t *new_buffer;
|
|
|
|
|
|
|
|
if ((new_buffer = malloc(sizeof(*new_buffer)))) {
|
|
|
|
memset(new_buffer, 0, sizeof(*new_buffer));
|
|
|
|
|
|
|
|
if (start_len) {
|
|
|
|
if (!(new_buffer->data = malloc(start_len))) {
|
|
|
|
free(new_buffer);
|
|
|
|
return SWITCH_STATUS_MEMERR;
|
|
|
|
}
|
|
|
|
memset(new_buffer->data, 0, start_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
new_buffer->max_len = max_len;
|
|
|
|
new_buffer->datalen = start_len;
|
|
|
|
new_buffer->id = buffer_id++;
|
|
|
|
new_buffer->blocksize = blocksize;
|
2006-09-21 19:52:56 +00:00
|
|
|
new_buffer->head = new_buffer->data;
|
2006-09-08 18:57:24 +00:00
|
|
|
switch_set_flag(new_buffer, SWITCH_BUFFER_FLAG_DYNAMIC);
|
|
|
|
|
|
|
|
*buffer = new_buffer;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_MEMERR;
|
|
|
|
}
|
|
|
|
|
2006-04-29 06:05:03 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_buffer_len(switch_buffer_t *buffer)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
|
|
|
|
|
|
|
assert(buffer != NULL);
|
|
|
|
|
2006-03-30 23:02:50 +00:00
|
|
|
return buffer->datalen;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-29 06:05:03 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_buffer_freespace(switch_buffer_t *buffer)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
|
|
|
assert(buffer != NULL);
|
|
|
|
|
2006-09-08 18:57:24 +00:00
|
|
|
if (switch_test_flag(buffer, SWITCH_BUFFER_FLAG_DYNAMIC)) {
|
|
|
|
if (buffer->max_len) {
|
|
|
|
return (switch_size_t) (buffer->max_len - buffer->used);
|
|
|
|
}
|
|
|
|
return 1000000;
|
|
|
|
}
|
|
|
|
|
2006-03-30 23:02:50 +00:00
|
|
|
return (switch_size_t) (buffer->datalen - buffer->used);
|
2006-09-08 18:57:24 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2006-04-29 06:05:03 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_buffer_inuse(switch_buffer_t *buffer)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
|
|
|
assert(buffer != NULL);
|
|
|
|
|
2006-03-30 23:02:50 +00:00
|
|
|
return buffer->used;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2006-04-29 06:05:03 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_buffer_toss(switch_buffer_t *buffer, switch_size_t datalen)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-03-30 23:02:50 +00:00
|
|
|
switch_size_t reading = 0;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
assert(buffer != NULL);
|
|
|
|
|
|
|
|
if (buffer->used < 1) {
|
|
|
|
buffer->used = 0;
|
|
|
|
return 0;
|
|
|
|
} else if (buffer->used >= datalen) {
|
|
|
|
reading = datalen;
|
|
|
|
} else {
|
|
|
|
reading = buffer->used;
|
|
|
|
}
|
|
|
|
|
2006-09-21 19:52:56 +00:00
|
|
|
memmove(buffer->data, buffer->data + reading, reading);
|
|
|
|
buffer->head = buffer->data;
|
|
|
|
buffer->used -= reading;
|
2006-09-22 02:10:27 +00:00
|
|
|
buffer->actually_used = buffer->used;
|
2006-09-21 19:52:56 +00:00
|
|
|
return buffer->used;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2006-04-29 06:05:03 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_buffer_read(switch_buffer_t *buffer, void *data, switch_size_t datalen)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-03-30 23:02:50 +00:00
|
|
|
switch_size_t reading = 0;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
assert(buffer != NULL);
|
|
|
|
assert(data != NULL);
|
|
|
|
|
|
|
|
|
|
|
|
if (buffer->used < 1) {
|
2005-12-06 21:25:56 +00:00
|
|
|
buffer->used = 0;
|
2005-11-19 20:07:43 +00:00
|
|
|
return 0;
|
|
|
|
} else if (buffer->used >= datalen) {
|
|
|
|
reading = datalen;
|
|
|
|
} else {
|
|
|
|
reading = buffer->used;
|
|
|
|
}
|
|
|
|
|
2006-09-21 19:52:56 +00:00
|
|
|
memcpy(data, buffer->head, reading);
|
2005-12-06 21:25:56 +00:00
|
|
|
buffer->used -= reading;
|
2006-09-21 19:52:56 +00:00
|
|
|
buffer->head += reading;
|
2006-09-20 14:22:32 +00:00
|
|
|
|
2006-09-21 19:52:56 +00:00
|
|
|
//if (buffer->id == 4) printf("%u o %d = %d\n", buffer->id, (uint32_t)reading, (uint32_t)buffer->used);
|
2006-03-30 23:02:50 +00:00
|
|
|
return reading;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
|
|
|
|
2006-04-29 06:05:03 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_buffer_write(switch_buffer_t *buffer, void *data, switch_size_t datalen)
|
2005-11-19 20:07:43 +00:00
|
|
|
{
|
2006-09-22 02:10:27 +00:00
|
|
|
switch_size_t freespace, actual_freespace;
|
2005-11-19 20:07:43 +00:00
|
|
|
|
|
|
|
assert(buffer != NULL);
|
|
|
|
assert(data != NULL);
|
|
|
|
assert(buffer->data != NULL);
|
2006-09-21 19:52:56 +00:00
|
|
|
|
|
|
|
if (!datalen) {
|
|
|
|
return buffer->used;
|
|
|
|
}
|
2006-09-08 18:57:24 +00:00
|
|
|
|
2006-09-22 02:10:27 +00:00
|
|
|
actual_freespace = buffer->datalen - buffer->actually_used;
|
|
|
|
|
|
|
|
if (actual_freespace < datalen) {
|
|
|
|
memmove(buffer->data, buffer->head, buffer->used);
|
|
|
|
buffer->head = buffer->data;
|
|
|
|
buffer->actually_used = buffer->used;
|
|
|
|
}
|
|
|
|
|
2006-09-08 18:57:24 +00:00
|
|
|
freespace = buffer->datalen - buffer->used;
|
|
|
|
|
2006-09-22 02:10:27 +00:00
|
|
|
/*
|
2006-09-21 19:52:56 +00:00
|
|
|
if (buffer->data != buffer->head) {
|
|
|
|
memmove(buffer->data, buffer->head, buffer->used);
|
|
|
|
buffer->head = buffer->data;
|
2006-09-14 00:15:03 +00:00
|
|
|
}
|
2006-09-22 02:10:27 +00:00
|
|
|
*/
|
2006-09-08 18:57:24 +00:00
|
|
|
if (switch_test_flag(buffer, SWITCH_BUFFER_FLAG_DYNAMIC)) {
|
|
|
|
if (freespace < datalen) {
|
|
|
|
switch_size_t new_size, new_block_size;
|
2006-09-21 19:52:56 +00:00
|
|
|
|
2006-09-08 18:57:24 +00:00
|
|
|
new_size = buffer->datalen + datalen;
|
|
|
|
new_block_size = buffer->datalen + buffer->blocksize;
|
|
|
|
|
|
|
|
if (new_block_size > new_size) {
|
|
|
|
new_size = new_block_size;
|
|
|
|
}
|
2006-09-21 19:52:56 +00:00
|
|
|
buffer->head = buffer->data;
|
2006-09-08 18:57:24 +00:00
|
|
|
if (!(buffer->data = realloc(buffer->data, new_size))) {
|
|
|
|
return 0;
|
|
|
|
}
|
2006-09-21 19:52:56 +00:00
|
|
|
buffer->head = buffer->data;
|
2006-09-08 18:57:24 +00:00
|
|
|
buffer->datalen = new_size;
|
|
|
|
}
|
|
|
|
}
|
2006-04-19 19:04:30 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
freespace = buffer->datalen - buffer->used;
|
2006-09-08 18:57:24 +00:00
|
|
|
|
2005-11-19 20:07:43 +00:00
|
|
|
if (freespace < datalen) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
2006-09-22 02:10:27 +00:00
|
|
|
memcpy(buffer->head + buffer->used, data, datalen);
|
2005-11-19 20:07:43 +00:00
|
|
|
buffer->used += datalen;
|
2006-09-22 02:10:27 +00:00
|
|
|
buffer->actually_used += datalen;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2006-09-21 19:52:56 +00:00
|
|
|
//if (buffer->id == 4) printf("%u i %d = %d\n", buffer->id, (uint32_t)datalen, (uint32_t)buffer->used);
|
2006-04-19 19:04:30 +00:00
|
|
|
|
2006-03-30 23:02:50 +00:00
|
|
|
return buffer->used;
|
2005-11-19 20:07:43 +00:00
|
|
|
}
|
2006-02-28 02:08:42 +00:00
|
|
|
|
2006-04-29 06:05:03 +00:00
|
|
|
SWITCH_DECLARE(void) switch_buffer_zero(switch_buffer_t *buffer)
|
2006-02-28 02:08:42 +00:00
|
|
|
{
|
|
|
|
assert(buffer != NULL);
|
|
|
|
assert(buffer->data != NULL);
|
|
|
|
|
|
|
|
buffer->used = 0;
|
2006-09-21 19:52:56 +00:00
|
|
|
buffer->head = buffer->data;
|
2006-02-28 02:08:42 +00:00
|
|
|
}
|
2006-09-08 18:57:24 +00:00
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_buffer_destroy(switch_buffer_t **buffer)
|
|
|
|
{
|
|
|
|
if (*buffer && switch_test_flag((*buffer), SWITCH_BUFFER_FLAG_DYNAMIC)) {
|
|
|
|
free((*buffer)->data);
|
|
|
|
free(*buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
*buffer = NULL;
|
|
|
|
}
|
2006-11-27 22:30:48 +00:00
|
|
|
|
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
|
|
|
* indent-tabs-mode:nil
|
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 expandtab:
|
|
|
|
*/
|