388 lines
10 KiB
C
388 lines
10 KiB
C
/*
|
|
* Asterisk -- An open source telephony toolkit.
|
|
*
|
|
* Copyright (C) 2014, Digium, Inc.
|
|
*
|
|
* Joshua Colp <jcolp@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 Media Format API
|
|
*
|
|
* \author Joshua Colp <jcolp@digium.com>
|
|
*/
|
|
|
|
/*** MODULEINFO
|
|
<support_level>core</support_level>
|
|
***/
|
|
|
|
#include "asterisk.h"
|
|
|
|
#include "asterisk/logger.h"
|
|
#include "asterisk/codec.h"
|
|
#include "asterisk/format.h"
|
|
#include "asterisk/astobj2.h"
|
|
#include "asterisk/strings.h"
|
|
#include "asterisk/module.h"
|
|
|
|
/*! \brief Number of buckets to use for format interfaces (should be prime for performance reasons) */
|
|
#define FORMAT_INTERFACE_BUCKETS 53
|
|
|
|
/*! \brief Definition of a media format */
|
|
struct ast_format {
|
|
/*! Name of the format */
|
|
const char *name;
|
|
/*! \brief Pointer to the codec in use for this format */
|
|
struct ast_codec *codec;
|
|
/*! \brief Attribute specific data, implementation specific */
|
|
void *attribute_data;
|
|
/*! \brief Pointer to the optional format interface */
|
|
const struct ast_format_interface *interface;
|
|
/*! \brief The number if audio channels used, if more than one an interleaved format is required */
|
|
unsigned int channel_count;
|
|
};
|
|
|
|
/*! \brief Structure used when registering a format interface */
|
|
struct format_interface {
|
|
/*! \brief Pointer to the format interface itself */
|
|
const struct ast_format_interface *interface;
|
|
/*! \brief Name of the codec the interface is for */
|
|
char codec[0];
|
|
};
|
|
|
|
/*! \brief Container for registered format interfaces */
|
|
static struct ao2_container *interfaces;
|
|
|
|
AO2_STRING_FIELD_HASH_FN(format_interface, codec)
|
|
AO2_STRING_FIELD_CMP_FN(format_interface, codec)
|
|
|
|
/*! \brief Function called when the process is shutting down */
|
|
static void format_shutdown(void)
|
|
{
|
|
ao2_cleanup(interfaces);
|
|
interfaces = NULL;
|
|
}
|
|
|
|
int ast_format_init(void)
|
|
{
|
|
interfaces = ao2_container_alloc_hash(AO2_ALLOC_OPT_LOCK_RWLOCK, 0,
|
|
FORMAT_INTERFACE_BUCKETS, format_interface_hash_fn, NULL, format_interface_cmp_fn);
|
|
if (!interfaces) {
|
|
return -1;
|
|
}
|
|
|
|
ast_register_cleanup(format_shutdown);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int __ast_format_interface_register(const char *codec, const struct ast_format_interface *interface, struct ast_module *mod)
|
|
{
|
|
SCOPED_AO2WRLOCK(lock, interfaces);
|
|
struct format_interface *format_interface;
|
|
|
|
if (!interface->format_clone || !interface->format_destroy) {
|
|
ast_log(LOG_ERROR, "Format interface for codec '%s' does not implement required callbacks\n", codec);
|
|
return -1;
|
|
}
|
|
|
|
format_interface = ao2_find(interfaces, codec, OBJ_SEARCH_KEY | OBJ_NOLOCK);
|
|
if (format_interface) {
|
|
ast_log(LOG_ERROR, "A format interface is already present for codec '%s'\n", codec);
|
|
ao2_ref(format_interface, -1);
|
|
return -1;
|
|
}
|
|
|
|
format_interface = ao2_alloc_options(sizeof(*format_interface) + strlen(codec) + 1,
|
|
NULL, AO2_ALLOC_OPT_LOCK_NOLOCK);
|
|
if (!format_interface) {
|
|
return -1;
|
|
}
|
|
format_interface->interface = interface;
|
|
strcpy(format_interface->codec, codec); /* Safe */
|
|
|
|
/* Once registered a format interface cannot be unregistered. */
|
|
ast_module_shutdown_ref(mod);
|
|
ao2_link_flags(interfaces, format_interface, OBJ_NOLOCK);
|
|
ao2_ref(format_interface, -1);
|
|
|
|
ast_verb(2, "Registered format interface for codec '%s'\n", codec);
|
|
|
|
return 0;
|
|
}
|
|
|
|
void *ast_format_get_attribute_data(const struct ast_format *format)
|
|
{
|
|
return format->attribute_data;
|
|
}
|
|
|
|
void ast_format_set_attribute_data(struct ast_format *format, void *attribute_data)
|
|
{
|
|
format->attribute_data = attribute_data;
|
|
}
|
|
|
|
unsigned int ast_format_get_channel_count(const struct ast_format *format)
|
|
{
|
|
return format->channel_count;
|
|
}
|
|
|
|
void ast_format_set_channel_count(struct ast_format *format, unsigned int channel_count)
|
|
{
|
|
format->channel_count = channel_count;
|
|
}
|
|
|
|
/*! \brief Destructor for media formats */
|
|
static void format_destroy(void *obj)
|
|
{
|
|
struct ast_format *format = obj;
|
|
|
|
if (format->interface) {
|
|
format->interface->format_destroy(format);
|
|
}
|
|
|
|
ao2_cleanup(format->codec);
|
|
}
|
|
|
|
struct ast_format *ast_format_create_named(const char *format_name, struct ast_codec *codec)
|
|
{
|
|
struct ast_format *format;
|
|
struct format_interface *format_interface;
|
|
|
|
format = ao2_t_alloc_options(sizeof(*format), format_destroy,
|
|
AO2_ALLOC_OPT_LOCK_NOLOCK, S_OR(codec->description, ""));
|
|
if (!format) {
|
|
return NULL;
|
|
}
|
|
format->name = format_name;
|
|
format->codec = ao2_bump(codec);
|
|
format->channel_count = 1;
|
|
|
|
format_interface = ao2_find(interfaces, codec->name, OBJ_SEARCH_KEY);
|
|
if (format_interface) {
|
|
format->interface = format_interface->interface;
|
|
ao2_ref(format_interface, -1);
|
|
}
|
|
|
|
return format;
|
|
}
|
|
|
|
struct ast_format *ast_format_clone(const struct ast_format *format)
|
|
{
|
|
struct ast_format *cloned = ast_format_create_named(format->name, format->codec);
|
|
|
|
if (!cloned) {
|
|
return NULL;
|
|
}
|
|
|
|
if (cloned->interface && cloned->interface->format_clone(format, cloned)) {
|
|
ao2_ref(cloned, -1);
|
|
return NULL;
|
|
}
|
|
|
|
return cloned;
|
|
}
|
|
|
|
struct ast_format *ast_format_create(struct ast_codec *codec)
|
|
{
|
|
return ast_format_create_named(codec->name, codec);
|
|
}
|
|
|
|
enum ast_format_cmp_res ast_format_cmp(const struct ast_format *format1, const struct ast_format *format2)
|
|
{
|
|
const struct ast_format_interface *interface;
|
|
|
|
if (format1 == NULL || format2 == NULL) {
|
|
return AST_FORMAT_CMP_NOT_EQUAL;
|
|
}
|
|
|
|
if (format1 == format2) {
|
|
return AST_FORMAT_CMP_EQUAL;
|
|
}
|
|
|
|
if (format1->codec != format2->codec) {
|
|
return AST_FORMAT_CMP_NOT_EQUAL;
|
|
}
|
|
|
|
interface = format1->interface ? format1->interface : format2->interface;
|
|
|
|
if (interface && interface->format_cmp) {
|
|
return interface->format_cmp(format1, format2);
|
|
}
|
|
|
|
return AST_FORMAT_CMP_EQUAL;
|
|
}
|
|
|
|
struct ast_format *ast_format_joint(const struct ast_format *format1, const struct ast_format *format2)
|
|
{
|
|
const struct ast_format_interface *interface;
|
|
|
|
if (format1->codec != format2->codec) {
|
|
return NULL;
|
|
}
|
|
|
|
/* If the two formats are the same structure OR if the codec is the same and no attributes
|
|
* exist we can immediately return a format with reference count bumped up, since they are
|
|
* the same.
|
|
*/
|
|
if ((ast_format_cmp(format1, format2) == AST_FORMAT_CMP_EQUAL && !format1->attribute_data && !format2->attribute_data)) {
|
|
return ao2_bump((struct ast_format*)format1);
|
|
}
|
|
|
|
interface = format1->interface ? format1->interface : format2->interface;
|
|
|
|
/* If there is attribute data on either there has to be an interface */
|
|
return interface->format_get_joint(format1, format2);
|
|
}
|
|
|
|
struct ast_format *ast_format_attribute_set(const struct ast_format *format, const char *name, const char *value)
|
|
{
|
|
const struct ast_format_interface *interface = format->interface;
|
|
|
|
if (!interface) {
|
|
struct format_interface *format_interface = ao2_find(interfaces, format->codec->name, OBJ_SEARCH_KEY);
|
|
if (format_interface) {
|
|
interface = format_interface->interface;
|
|
ao2_ref(format_interface, -1);
|
|
}
|
|
}
|
|
|
|
if (!interface || !interface->format_attribute_set) {
|
|
return ao2_bump((struct ast_format*)format);
|
|
}
|
|
|
|
return interface->format_attribute_set(format, name, value);
|
|
}
|
|
|
|
const void *ast_format_attribute_get(const struct ast_format *format, const char *name)
|
|
{
|
|
const struct ast_format_interface *interface = format->interface;
|
|
|
|
if (!interface) {
|
|
struct format_interface *format_interface = ao2_find(interfaces, format->codec->name, OBJ_SEARCH_KEY);
|
|
if (format_interface) {
|
|
interface = format_interface->interface;
|
|
ao2_ref(format_interface, -1);
|
|
}
|
|
}
|
|
|
|
if (!interface || !interface->format_attribute_get) {
|
|
return NULL;
|
|
}
|
|
|
|
return interface->format_attribute_get(format, name);
|
|
}
|
|
|
|
struct ast_format *ast_format_parse_sdp_fmtp(const struct ast_format *format, const char *attributes)
|
|
{
|
|
const struct ast_format_interface *interface = format->interface;
|
|
|
|
if (!interface) {
|
|
struct format_interface *format_interface = ao2_find(interfaces, format->codec->name, OBJ_SEARCH_KEY);
|
|
if (format_interface) {
|
|
interface = format_interface->interface;
|
|
ao2_ref(format_interface, -1);
|
|
}
|
|
}
|
|
|
|
if (!interface || !interface->format_parse_sdp_fmtp) {
|
|
return ao2_bump((struct ast_format*)format);
|
|
}
|
|
|
|
return interface->format_parse_sdp_fmtp(format, attributes);
|
|
}
|
|
|
|
void ast_format_generate_sdp_fmtp(const struct ast_format *format, unsigned int payload, struct ast_str **str)
|
|
{
|
|
const struct ast_format_interface *interface = format->interface;
|
|
|
|
if (!interface) {
|
|
struct format_interface *format_interface = ao2_find(interfaces, format->codec->name, OBJ_SEARCH_KEY);
|
|
if (format_interface) {
|
|
interface = format_interface->interface;
|
|
ao2_ref(format_interface, -1);
|
|
}
|
|
}
|
|
|
|
if (!interface || !interface->format_generate_sdp_fmtp) {
|
|
return;
|
|
}
|
|
|
|
interface->format_generate_sdp_fmtp(format, payload, str);
|
|
}
|
|
|
|
struct ast_codec *ast_format_get_codec(const struct ast_format *format)
|
|
{
|
|
return ao2_bump(format->codec);
|
|
}
|
|
|
|
unsigned int ast_format_get_codec_id(const struct ast_format *format)
|
|
{
|
|
return format->codec->id;
|
|
}
|
|
|
|
const char *ast_format_get_name(const struct ast_format *format)
|
|
{
|
|
return format->name;
|
|
}
|
|
|
|
const char *ast_format_get_codec_name(const struct ast_format *format)
|
|
{
|
|
return format->codec->name;
|
|
}
|
|
|
|
int ast_format_can_be_smoothed(const struct ast_format *format)
|
|
{
|
|
return format->codec->smooth;
|
|
}
|
|
|
|
int ast_format_get_smoother_flags(const struct ast_format *format)
|
|
{
|
|
return format->codec->smoother_flags;
|
|
}
|
|
|
|
enum ast_media_type ast_format_get_type(const struct ast_format *format)
|
|
{
|
|
return format->codec->type;
|
|
}
|
|
|
|
unsigned int ast_format_get_default_ms(const struct ast_format *format)
|
|
{
|
|
return format->codec->default_ms;
|
|
}
|
|
|
|
unsigned int ast_format_get_minimum_ms(const struct ast_format *format)
|
|
{
|
|
return format->codec->minimum_ms;
|
|
}
|
|
|
|
unsigned int ast_format_get_maximum_ms(const struct ast_format *format)
|
|
{
|
|
return format->codec->maximum_ms;
|
|
}
|
|
|
|
unsigned int ast_format_get_minimum_bytes(const struct ast_format *format)
|
|
{
|
|
return format->codec->minimum_bytes;
|
|
}
|
|
|
|
unsigned int ast_format_get_sample_rate(const struct ast_format *format)
|
|
{
|
|
return format->codec->sample_rate ?: 8000;
|
|
}
|
|
|
|
unsigned int ast_format_determine_length(const struct ast_format *format, unsigned int samples)
|
|
{
|
|
return ast_codec_determine_length(format->codec, samples);
|
|
}
|