2007-10-25 09:38:15 +00:00
|
|
|
/* codecs.c
|
|
|
|
* codecs interface 2007 Tomas Kukosa
|
|
|
|
*
|
|
|
|
* Wireshark - Network traffic analyzer
|
|
|
|
* By Gerald Combs <gerald@wireshark.org>
|
|
|
|
* Copyright 1998 Gerald Combs
|
2013-12-13 14:19:52 +00:00
|
|
|
*
|
2007-10-25 09:38:15 +00:00
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
2013-12-13 14:19:52 +00:00
|
|
|
*
|
2007-10-25 09:38:15 +00:00
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
2013-12-13 14:19:52 +00:00
|
|
|
*
|
2007-10-25 09:38:15 +00:00
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
2012-06-28 22:56:06 +00:00
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
2007-10-25 09:38:15 +00:00
|
|
|
*/
|
|
|
|
|
2013-12-02 08:30:29 +00:00
|
|
|
#include "config.h"
|
|
|
|
|
2013-12-01 03:53:57 +00:00
|
|
|
#include <glib.h>
|
|
|
|
#include "codecs.h"
|
2012-09-14 13:08:17 +00:00
|
|
|
|
|
|
|
#include "G711a/G711adecode.h"
|
|
|
|
#include "G711u/G711udecode.h"
|
|
|
|
|
|
|
|
#ifdef HAVE_SBC
|
|
|
|
#include "sbc/sbc_private.h"
|
|
|
|
#endif
|
2007-10-25 09:38:15 +00:00
|
|
|
|
2016-11-23 23:27:14 +00:00
|
|
|
#ifdef HAVE_SPANDSP
|
|
|
|
#include "G722/G722decode.h"
|
|
|
|
#include "G726/G726decode.h"
|
|
|
|
#endif
|
|
|
|
|
2017-07-26 21:23:12 +00:00
|
|
|
#ifdef HAVE_BCG729
|
|
|
|
#include "G729/G729decode.h"
|
|
|
|
#endif
|
|
|
|
|
2013-12-02 08:30:29 +00:00
|
|
|
#ifdef HAVE_PLUGINS
|
|
|
|
|
|
|
|
#include <gmodule.h>
|
|
|
|
|
|
|
|
#include <wsutil/plugins.h>
|
|
|
|
|
|
|
|
/*
|
|
|
|
* List of codec plugins.
|
|
|
|
*/
|
|
|
|
typedef struct {
|
2012-09-14 13:08:17 +00:00
|
|
|
void (*register_codec_module)(void); /* routine to call to register a codec */
|
2013-12-02 08:30:29 +00:00
|
|
|
} codec_plugin;
|
|
|
|
|
|
|
|
static GSList *codec_plugins = NULL;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Callback for each plugin found.
|
|
|
|
*/
|
|
|
|
static gboolean
|
|
|
|
check_for_codec_plugin(GModule *handle)
|
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
gpointer gp;
|
|
|
|
void (*register_codec_module)(void);
|
|
|
|
codec_plugin *plugin;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Do we have a register_codec_module routine?
|
|
|
|
*/
|
|
|
|
if (!g_module_symbol(handle, "register_codec_module", &gp)) {
|
|
|
|
/* No, so this isn't a codec plugin. */
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Yes - this plugin includes one or more codecs.
|
|
|
|
*/
|
|
|
|
register_codec_module = (void (*)(void))gp;
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Add this one to the list of codec plugins.
|
|
|
|
*/
|
|
|
|
plugin = (codec_plugin *)g_malloc(sizeof (codec_plugin));
|
|
|
|
plugin->register_codec_module = register_codec_module;
|
|
|
|
codec_plugins = g_slist_append(codec_plugins, plugin);
|
|
|
|
return TRUE;
|
2013-12-02 08:30:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
codec_register_plugin_types(void)
|
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
add_plugin_type("codec", check_for_codec_plugin);
|
2013-12-02 08:30:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
register_codec_plugin(gpointer data, gpointer user_data _U_)
|
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
codec_plugin *plugin = (codec_plugin *)data;
|
2013-12-02 08:30:29 +00:00
|
|
|
|
2012-09-14 13:08:17 +00:00
|
|
|
(plugin->register_codec_module)();
|
2013-12-02 08:30:29 +00:00
|
|
|
}
|
2016-11-28 23:53:23 +00:00
|
|
|
#endif /* HAVE_PLUGINS */
|
2013-12-02 08:30:29 +00:00
|
|
|
|
2015-09-04 15:12:25 +00:00
|
|
|
|
2013-12-02 08:30:29 +00:00
|
|
|
/*
|
|
|
|
* For all codec plugins, call their register routines.
|
|
|
|
*/
|
|
|
|
void
|
|
|
|
register_all_codecs(void)
|
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
register_codec("g711U", codec_g711u_init, codec_g711u_release,
|
|
|
|
codec_g711u_get_channels, codec_g711u_get_frequency, codec_g711u_decode);
|
|
|
|
register_codec("g711A", codec_g711a_init, codec_g711a_release,
|
|
|
|
codec_g711a_get_channels, codec_g711a_get_frequency, codec_g711a_decode);
|
|
|
|
#ifdef HAVE_SPANDSP
|
|
|
|
register_codec("g722", codec_g722_init, codec_g722_release,
|
|
|
|
codec_g722_get_channels, codec_g722_get_frequency, codec_g722_decode);
|
2016-11-23 23:27:14 +00:00
|
|
|
register_codec("G726-16", codec_g726_16_init, codec_g726_release,
|
|
|
|
codec_g726_get_channels, codec_g726_get_frequency, codec_g726_decode);
|
|
|
|
register_codec("G726-24", codec_g726_24_init, codec_g726_release,
|
|
|
|
codec_g726_get_channels, codec_g726_get_frequency, codec_g726_decode);
|
|
|
|
register_codec("G726-32", codec_g726_32_init, codec_g726_release,
|
|
|
|
codec_g726_get_channels, codec_g726_get_frequency, codec_g726_decode);
|
|
|
|
register_codec("G726-40", codec_g726_40_init, codec_g726_release,
|
|
|
|
codec_g726_get_channels, codec_g726_get_frequency, codec_g726_decode);
|
|
|
|
register_codec("AAL2-G726-16", codec_aal2_g726_16_init, codec_g726_release,
|
|
|
|
codec_g726_get_channels, codec_g726_get_frequency, codec_g726_decode);
|
|
|
|
register_codec("AAL2-G726-24", codec_aal2_g726_24_init, codec_g726_release,
|
|
|
|
codec_g726_get_channels, codec_g726_get_frequency, codec_g726_decode);
|
|
|
|
register_codec("AAL2-G726-32", codec_aal2_g726_32_init, codec_g726_release,
|
|
|
|
codec_g726_get_channels, codec_g726_get_frequency, codec_g726_decode);
|
|
|
|
register_codec("AAL2-G726-40", codec_aal2_g726_40_init, codec_g726_release,
|
2012-09-14 13:08:17 +00:00
|
|
|
codec_g726_get_channels, codec_g726_get_frequency, codec_g726_decode);
|
|
|
|
#endif
|
2017-07-26 21:23:12 +00:00
|
|
|
#ifdef HAVE_BCG729
|
|
|
|
register_codec("g729", codec_g729_init, codec_g729_release,
|
|
|
|
codec_g729_get_channels, codec_g729_get_frequency, codec_g729_decode);
|
|
|
|
#endif
|
|
|
|
#ifdef HAVE_SBC
|
|
|
|
register_codec("SBC", codec_sbc_init, codec_sbc_release,
|
|
|
|
codec_sbc_get_channels, codec_sbc_get_frequency, codec_sbc_decode);
|
|
|
|
#endif
|
2012-09-14 13:08:17 +00:00
|
|
|
#ifdef HAVE_SBC
|
|
|
|
register_codec("SBC", codec_sbc_init, codec_sbc_release,
|
|
|
|
codec_sbc_get_channels, codec_sbc_get_frequency, codec_sbc_decode);
|
|
|
|
#endif
|
|
|
|
|
2016-11-28 23:53:23 +00:00
|
|
|
#ifdef HAVE_PLUGINS
|
2012-09-14 13:08:17 +00:00
|
|
|
g_slist_foreach(codec_plugins, register_codec_plugin, NULL);
|
2013-12-02 08:30:29 +00:00
|
|
|
#endif /* HAVE_PLUGINS */
|
2016-11-28 23:53:23 +00:00
|
|
|
}
|
|
|
|
|
2013-12-02 08:30:29 +00:00
|
|
|
|
2007-10-25 09:38:15 +00:00
|
|
|
struct codec_handle {
|
2012-09-14 13:08:17 +00:00
|
|
|
const char *name;
|
|
|
|
codec_init_fn init_fn;
|
|
|
|
codec_release_fn release_fn;
|
|
|
|
codec_get_channels_fn channels_fn;
|
|
|
|
codec_get_frequency_fn frequency_fn;
|
|
|
|
codec_decode_fn decode_fn;
|
2007-10-25 09:38:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* List of registered codecs.
|
|
|
|
*/
|
|
|
|
static GHashTable *registered_codecs = NULL;
|
|
|
|
|
|
|
|
|
|
|
|
/* Find a registered codec by name. */
|
|
|
|
codec_handle_t
|
|
|
|
find_codec(const char *name)
|
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
return (registered_codecs) ? (codec_handle_t)g_hash_table_lookup(registered_codecs, name) : NULL;
|
2007-10-25 09:38:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Register a codec by name. */
|
2013-12-02 08:51:46 +00:00
|
|
|
gboolean
|
2012-09-14 13:08:17 +00:00
|
|
|
register_codec(const char *name, codec_init_fn init_fn, codec_release_fn release_fn,
|
|
|
|
codec_get_channels_fn channels_fn, codec_get_frequency_fn frequency_fn,
|
|
|
|
codec_decode_fn decode_fn)
|
2007-10-25 09:38:15 +00:00
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
struct codec_handle *handle;
|
|
|
|
|
|
|
|
/* Create our hash table if it doesn't already exist */
|
|
|
|
if (registered_codecs == NULL)
|
|
|
|
registered_codecs = g_hash_table_new(g_str_hash, g_str_equal);
|
|
|
|
|
|
|
|
/* Make sure the registration is unique */
|
|
|
|
if (g_hash_table_lookup(registered_codecs, name) != NULL)
|
|
|
|
return FALSE; /* report an error, or have our caller do it? */
|
|
|
|
|
|
|
|
handle = (struct codec_handle *)g_malloc(sizeof (struct codec_handle));
|
|
|
|
handle->name = name;
|
|
|
|
handle->init_fn = init_fn;
|
|
|
|
handle->release_fn = release_fn;
|
|
|
|
handle->channels_fn = channels_fn;
|
|
|
|
handle->frequency_fn = frequency_fn;
|
|
|
|
handle->decode_fn = decode_fn;
|
|
|
|
|
|
|
|
g_hash_table_insert(registered_codecs, (gpointer)name, (gpointer) handle);
|
|
|
|
return TRUE;
|
2007-10-25 09:38:15 +00:00
|
|
|
}
|
|
|
|
|
2016-01-28 10:16:03 +00:00
|
|
|
/* Deregister a codec by name. */
|
|
|
|
gboolean
|
|
|
|
deregister_codec(const char *name)
|
|
|
|
{
|
|
|
|
gpointer key, value;
|
|
|
|
|
|
|
|
if (registered_codecs && g_hash_table_lookup_extended(registered_codecs, name, &key, &value)) {
|
|
|
|
g_hash_table_remove(registered_codecs, name);
|
|
|
|
g_free(value);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2007-10-25 09:38:15 +00:00
|
|
|
void *codec_init(codec_handle_t codec)
|
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
if (!codec) return NULL;
|
|
|
|
return (codec->init_fn)();
|
2007-10-25 09:38:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void codec_release(codec_handle_t codec, void *context)
|
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
if (!codec) return;
|
|
|
|
(codec->release_fn)(context);
|
|
|
|
}
|
|
|
|
|
2015-09-04 15:12:25 +00:00
|
|
|
unsigned codec_get_channels(codec_handle_t codec, void *context)
|
2012-09-14 13:08:17 +00:00
|
|
|
{
|
|
|
|
if (!codec) return 0;
|
|
|
|
return (codec->channels_fn)(context);
|
|
|
|
}
|
|
|
|
|
2015-09-04 15:12:25 +00:00
|
|
|
unsigned codec_get_frequency(codec_handle_t codec, void *context)
|
2012-09-14 13:08:17 +00:00
|
|
|
{
|
|
|
|
if (!codec) return 0;
|
|
|
|
return (codec->frequency_fn)(context);
|
2007-10-25 09:38:15 +00:00
|
|
|
}
|
|
|
|
|
2015-09-04 15:12:25 +00:00
|
|
|
size_t codec_decode(codec_handle_t codec, void *context, const void *input, size_t inputSizeBytes, void *output, size_t *outputSizeBytes)
|
2007-10-25 09:38:15 +00:00
|
|
|
{
|
2012-09-14 13:08:17 +00:00
|
|
|
if (!codec) return 0;
|
|
|
|
return (codec->decode_fn)(context, input, inputSizeBytes, output, outputSizeBytes);
|
2007-10-25 09:38:15 +00:00
|
|
|
}
|
2012-09-14 13:08:17 +00:00
|
|
|
|
2016-11-28 23:58:40 +00:00
|
|
|
/**
|
|
|
|
* Get compile-time information for libraries used by libwscodecs.
|
|
|
|
*/
|
|
|
|
void codec_get_compiled_version_info(GString *str)
|
|
|
|
{
|
|
|
|
/* SBC */
|
|
|
|
#ifdef HAVE_SBC
|
|
|
|
g_string_append(str, ", with SBC");
|
|
|
|
#else
|
|
|
|
g_string_append(str, ", without SBC");
|
|
|
|
#endif
|
|
|
|
|
2016-12-06 19:49:12 +00:00
|
|
|
/* SpanDSP (G.722, G.726) */
|
2016-11-28 23:58:40 +00:00
|
|
|
#ifdef HAVE_SPANDSP
|
2016-12-06 19:49:12 +00:00
|
|
|
g_string_append(str, ", with SpanDSP");
|
2016-11-28 23:58:40 +00:00
|
|
|
#else
|
2016-12-06 19:49:12 +00:00
|
|
|
g_string_append(str, ", without SpanDSP");
|
2016-11-28 23:58:40 +00:00
|
|
|
#endif
|
2017-07-26 21:23:12 +00:00
|
|
|
|
|
|
|
/* BCG729 (G.729) */
|
|
|
|
#ifdef HAVE_BCG729
|
|
|
|
g_string_append(str, ", with bcg729");
|
|
|
|
#else
|
|
|
|
g_string_append(str, ", without bcg729");
|
|
|
|
#endif
|
2016-11-28 23:58:40 +00:00
|
|
|
}
|
|
|
|
|
2012-09-14 13:08:17 +00:00
|
|
|
/*
|
|
|
|
* Editor modelines - http://www.wireshark.org/tools/modelines.html
|
|
|
|
*
|
|
|
|
* Local variables:
|
|
|
|
* c-basic-offset: 4
|
|
|
|
* tab-width: 8
|
|
|
|
* indent-tabs-mode: nil
|
|
|
|
* End:
|
|
|
|
*
|
|
|
|
* vi: set shiftwidth=4 tabstop=8 expandtab:
|
|
|
|
* :indentSize=4:tabSize=8:noTabs=true:
|
|
|
|
*/
|