2007-03-29 22:34:40 +00:00
|
|
|
/*
|
|
|
|
* FreeSWITCH Modular Media Switching Software Library / Soft-Switch Application
|
2014-02-05 21:02:28 +00:00
|
|
|
* Copyright (C) 2005-2014, Anthony Minessale II <anthm@freeswitch.org>
|
2007-03-29 22:34:40 +00:00
|
|
|
*
|
|
|
|
* 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
|
2009-02-04 21:20:54 +00:00
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
2007-03-29 22:34:40 +00:00
|
|
|
* Portions created by the Initial Developer are Copyright (C)
|
|
|
|
* the Initial Developer. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Contributor(s):
|
|
|
|
*
|
2009-02-04 21:20:54 +00:00
|
|
|
* Anthony Minessale II <anthm@freeswitch.org>
|
2007-03-29 22:34:40 +00:00
|
|
|
* Michael Jerris <mike@jerris.com>
|
|
|
|
* Paul D. Tinsley <pdt at jackhammer.org>
|
2012-06-11 13:05:29 +00:00
|
|
|
* Christopher M. Rienzo <chris@rienzo.com>
|
2007-03-29 22:34:40 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* switch_core_codec.c -- Main Core Library (codec functions)
|
|
|
|
*
|
|
|
|
*/
|
2008-01-27 17:36:53 +00:00
|
|
|
|
2007-03-29 22:34:40 +00:00
|
|
|
#include <switch.h>
|
2007-05-14 17:10:46 +00:00
|
|
|
#include "private/switch_core_pvt.h"
|
2007-03-29 22:34:40 +00:00
|
|
|
|
2010-03-17 16:22:12 +00:00
|
|
|
static uint32_t CODEC_ID = 1;
|
2008-03-27 22:22:46 +00:00
|
|
|
|
|
|
|
SWITCH_DECLARE(uint32_t) switch_core_codec_next_id(void)
|
|
|
|
{
|
|
|
|
return CODEC_ID++;
|
|
|
|
}
|
|
|
|
|
2008-07-15 17:39:16 +00:00
|
|
|
SWITCH_DECLARE(void) switch_core_session_unset_read_codec(switch_core_session_t *session)
|
|
|
|
{
|
2008-11-19 20:12:20 +00:00
|
|
|
switch_mutex_t *mutex = NULL;
|
|
|
|
|
2008-11-21 00:09:11 +00:00
|
|
|
switch_mutex_lock(session->codec_read_mutex);
|
2010-07-06 18:20:27 +00:00
|
|
|
if (session->read_codec) mutex = session->read_codec->mutex;
|
|
|
|
if (mutex) switch_mutex_lock(mutex);
|
2008-07-15 17:39:16 +00:00
|
|
|
session->real_read_codec = session->read_codec = NULL;
|
2008-10-29 15:57:57 +00:00
|
|
|
session->raw_read_frame.codec = session->read_codec;
|
|
|
|
session->raw_write_frame.codec = session->read_codec;
|
|
|
|
session->enc_read_frame.codec = session->read_codec;
|
|
|
|
session->enc_write_frame.codec = session->read_codec;
|
2010-07-06 18:20:27 +00:00
|
|
|
if (mutex) switch_mutex_unlock(mutex);
|
2008-11-21 00:09:11 +00:00
|
|
|
switch_mutex_unlock(session->codec_read_mutex);
|
2008-07-15 17:39:16 +00:00
|
|
|
}
|
|
|
|
|
2008-12-18 23:07:32 +00:00
|
|
|
SWITCH_DECLARE(void) switch_core_session_lock_codec_write(switch_core_session_t *session)
|
|
|
|
{
|
|
|
|
switch_mutex_lock(session->codec_write_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_core_session_unlock_codec_write(switch_core_session_t *session)
|
|
|
|
{
|
|
|
|
switch_mutex_unlock(session->codec_write_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_core_session_lock_codec_read(switch_core_session_t *session)
|
|
|
|
{
|
|
|
|
switch_mutex_lock(session->codec_read_mutex);
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_core_session_unlock_codec_read(switch_core_session_t *session)
|
|
|
|
{
|
|
|
|
switch_mutex_unlock(session->codec_read_mutex);
|
|
|
|
}
|
|
|
|
|
2008-10-14 18:03:14 +00:00
|
|
|
SWITCH_DECLARE(void) switch_core_session_unset_write_codec(switch_core_session_t *session)
|
2010-02-06 03:38:24 +00:00
|
|
|
{
|
2008-11-19 20:12:20 +00:00
|
|
|
switch_mutex_t *mutex = NULL;
|
|
|
|
|
2008-11-21 00:09:11 +00:00
|
|
|
switch_mutex_lock(session->codec_write_mutex);
|
2010-07-06 18:20:27 +00:00
|
|
|
if (session->write_codec) mutex = session->write_codec->mutex;
|
|
|
|
if (mutex) switch_mutex_lock(mutex);
|
2008-10-14 18:03:14 +00:00
|
|
|
session->real_write_codec = session->write_codec = NULL;
|
2010-07-06 18:20:27 +00:00
|
|
|
if (mutex) switch_mutex_unlock(mutex);
|
2008-11-21 00:09:11 +00:00
|
|
|
switch_mutex_unlock(session->codec_write_mutex);
|
2008-10-14 18:03:14 +00:00
|
|
|
}
|
|
|
|
|
2012-01-17 15:43:14 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_real_read_codec(switch_core_session_t *session, switch_codec_t *codec)
|
|
|
|
{
|
|
|
|
switch_event_t *event;
|
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
char tmp[30];
|
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
|
|
|
int changed_read_codec = 0;
|
|
|
|
|
|
|
|
switch_mutex_lock(session->codec_read_mutex);
|
|
|
|
|
|
|
|
if (codec && (!codec->implementation || !switch_core_codec_ready(codec))) {
|
|
|
|
codec = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (codec) {
|
|
|
|
/* set real_read_codec and read_codec */
|
|
|
|
if (!session->real_read_codec) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Original read codec set to %s:%d\n",
|
|
|
|
switch_channel_get_name(session->channel), codec->implementation->iananame, codec->implementation->ianacode);
|
|
|
|
session->read_codec = session->real_read_codec = codec;
|
2012-01-17 17:00:43 +00:00
|
|
|
changed_read_codec = 1;
|
2012-01-17 15:43:14 +00:00
|
|
|
if (codec->implementation) {
|
|
|
|
session->read_impl = *codec->implementation;
|
2012-05-14 18:07:35 +00:00
|
|
|
session->real_read_impl = *codec->implementation;
|
2012-01-17 15:43:14 +00:00
|
|
|
} else {
|
|
|
|
memset(&session->read_impl, 0, sizeof(session->read_impl));
|
|
|
|
}
|
|
|
|
} else { /* replace real_read_codec */
|
|
|
|
switch_codec_t *cur_codec;
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Original read codec replaced with %s:%d\n",
|
|
|
|
switch_channel_get_name(session->channel), codec->implementation->iananame, codec->implementation->ianacode);
|
|
|
|
/* Set real_read_codec to front of the list of read_codecs */
|
|
|
|
cur_codec = session->read_codec;
|
|
|
|
while (cur_codec != NULL) {
|
|
|
|
if (cur_codec->next == session->real_read_codec) {
|
|
|
|
cur_codec->next = codec;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
cur_codec = cur_codec->next;
|
|
|
|
}
|
|
|
|
session->real_read_codec = codec;
|
|
|
|
/* set read_codec with real_read_codec if it no longer is ready */
|
|
|
|
if (!switch_core_codec_ready(session->read_codec)) {
|
|
|
|
session->read_codec = codec;
|
|
|
|
changed_read_codec = 1;
|
|
|
|
if (codec->implementation) {
|
2012-05-14 18:07:35 +00:00
|
|
|
session->read_impl = *codec->implementation;
|
|
|
|
session->real_read_impl = *codec->implementation;
|
2012-01-17 15:43:14 +00:00
|
|
|
} else {
|
|
|
|
memset(&session->read_impl, 0, sizeof(session->read_impl));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2012-01-20 19:30:53 +00:00
|
|
|
|
|
|
|
/* force media bugs to copy the read codec from the next frame */
|
|
|
|
switch_thread_rwlock_wrlock(session->bug_rwlock);
|
|
|
|
if (switch_core_codec_ready(&session->bug_codec)) {
|
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Destroying BUG Codec %s:%d\n",
|
|
|
|
session->bug_codec.implementation->iananame, session->bug_codec.implementation->ianacode);
|
|
|
|
switch_core_codec_destroy(&session->bug_codec);
|
|
|
|
}
|
|
|
|
switch_thread_rwlock_unlock(session->bug_rwlock);
|
2012-01-17 15:43:14 +00:00
|
|
|
} else {
|
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (changed_read_codec && session->read_codec && session->read_impl.decoded_bytes_per_packet) {
|
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_CODEC) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_channel_event_set_data(session->channel, event);
|
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "channel-read-codec-name", session->read_impl.iananame);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-read-codec-rate", "%d", session->read_impl.actual_samples_per_second);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-read-codec-bit-rate", "%d", session->read_impl.bits_per_second);
|
|
|
|
if (session->read_impl.actual_samples_per_second != session->read_impl.samples_per_second) {
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-reported-read-codec-rate", "%d", session->read_impl.samples_per_second);
|
|
|
|
}
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_channel_set_variable(channel, "read_codec", session->read_impl.iananame);
|
2013-10-15 21:24:32 +00:00
|
|
|
switch_channel_set_variable(channel, "original_read_codec", session->read_impl.iananame);
|
2012-01-17 15:43:14 +00:00
|
|
|
switch_snprintf(tmp, sizeof(tmp), "%d", session->read_impl.actual_samples_per_second);
|
|
|
|
switch_channel_set_variable(channel, "read_rate", tmp);
|
2013-10-15 21:24:32 +00:00
|
|
|
switch_channel_set_variable(channel, "original_read_rate", tmp);
|
2012-01-17 15:43:14 +00:00
|
|
|
|
|
|
|
session->raw_read_frame.codec = session->read_codec;
|
|
|
|
session->raw_write_frame.codec = session->read_codec;
|
|
|
|
session->enc_read_frame.codec = session->read_codec;
|
|
|
|
session->enc_write_frame.codec = session->read_codec;
|
|
|
|
}
|
|
|
|
|
|
|
|
end:
|
|
|
|
|
|
|
|
if (session->read_codec) {
|
|
|
|
switch_channel_set_flag(channel, CF_MEDIA_SET);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_mutex_unlock(session->codec_read_mutex);
|
|
|
|
return status;
|
|
|
|
}
|
2008-03-27 22:22:46 +00:00
|
|
|
|
2007-03-30 00:13:31 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_read_codec(switch_core_session_t *session, switch_codec_t *codec)
|
2007-03-29 22:34:40 +00:00
|
|
|
{
|
|
|
|
switch_event_t *event;
|
2008-01-28 07:26:10 +00:00
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
2007-12-04 18:52:13 +00:00
|
|
|
char tmp[30];
|
2008-10-15 21:54:26 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2008-11-21 00:09:11 +00:00
|
|
|
switch_mutex_lock(session->codec_read_mutex);
|
|
|
|
|
2009-04-09 17:17:12 +00:00
|
|
|
if (codec && (!codec->implementation || !switch_core_codec_ready(codec))) {
|
2008-10-29 15:57:57 +00:00
|
|
|
codec = NULL;
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2008-10-29 15:57:57 +00:00
|
|
|
if (codec) {
|
|
|
|
if (!session->real_read_codec) {
|
|
|
|
session->read_codec = session->real_read_codec = codec;
|
2010-02-26 19:20:24 +00:00
|
|
|
if (codec->implementation) {
|
|
|
|
session->read_impl = *codec->implementation;
|
2012-05-14 15:03:35 +00:00
|
|
|
session->real_read_impl = *codec->implementation;
|
2010-02-26 19:20:24 +00:00
|
|
|
} else {
|
|
|
|
memset(&session->read_impl, 0, sizeof(session->read_impl));
|
|
|
|
}
|
2008-10-29 15:57:57 +00:00
|
|
|
} else {
|
2012-01-18 20:11:46 +00:00
|
|
|
if (codec == session->read_codec) {
|
|
|
|
goto end;
|
|
|
|
}
|
2009-12-18 23:19:53 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Push codec %s:%d\n",
|
|
|
|
switch_channel_get_name(session->channel), codec->implementation->iananame, codec->implementation->ianacode);
|
|
|
|
codec->next = session->read_codec;
|
2008-10-29 15:57:57 +00:00
|
|
|
session->read_codec = codec;
|
2010-02-26 19:20:24 +00:00
|
|
|
if (codec->implementation) {
|
|
|
|
session->read_impl = *codec->implementation;
|
|
|
|
} else {
|
|
|
|
memset(&session->read_impl, 0, sizeof(session->read_impl));
|
|
|
|
}
|
2008-10-29 15:57:57 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (session->read_codec == session->real_read_codec) {
|
|
|
|
goto end;
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-12-18 23:19:53 +00:00
|
|
|
if (session->read_codec->next) {
|
|
|
|
switch_codec_t *old = session->read_codec;
|
|
|
|
session->read_codec = session->read_codec->next;
|
2010-02-26 19:20:24 +00:00
|
|
|
if (session->read_codec->implementation) {
|
|
|
|
session->read_impl = *session->read_codec->implementation;
|
|
|
|
} else {
|
|
|
|
memset(&session->read_impl, 0, sizeof(session->read_impl));
|
|
|
|
}
|
2009-12-18 23:19:53 +00:00
|
|
|
old->next = NULL;
|
2010-10-06 20:16:09 +00:00
|
|
|
|
2009-12-18 23:19:53 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "%s Restore previous codec %s:%d.\n",
|
2010-02-06 03:38:24 +00:00
|
|
|
switch_channel_get_name(session->channel),
|
2010-10-06 20:16:09 +00:00
|
|
|
session->read_impl.iananame ? session->read_impl.iananame : "N/A", session->read_impl.ianacode);
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2009-12-18 23:19:53 +00:00
|
|
|
} else if (session->real_read_codec) {
|
2008-10-29 00:04:20 +00:00
|
|
|
session->read_codec = session->real_read_codec;
|
2010-02-26 19:20:24 +00:00
|
|
|
if (session->real_read_codec->implementation) {
|
|
|
|
session->read_impl = *session->real_read_codec->implementation;
|
|
|
|
} else {
|
|
|
|
memset(&session->read_impl, 0, sizeof(session->read_impl));
|
|
|
|
}
|
2009-08-13 20:35:02 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_DEBUG, "Restore original codec.\n");
|
2008-07-09 16:44:19 +00:00
|
|
|
} else {
|
2008-10-15 21:54:26 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
goto end;
|
2008-07-09 16:44:19 +00:00
|
|
|
}
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2008-10-29 00:04:20 +00:00
|
|
|
if (!session->read_codec) {
|
2010-02-06 03:38:24 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
goto end;
|
2008-10-29 00:04:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (session->read_codec && session->read_impl.decoded_bytes_per_packet) {
|
2008-07-15 17:39:16 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_CODEC) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_channel_event_set_data(session->channel, event);
|
2008-10-16 16:34:52 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "channel-read-codec-name", session->read_impl.iananame);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-read-codec-rate", "%d", session->read_impl.actual_samples_per_second);
|
2010-09-29 21:52:34 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-read-codec-bit-rate", "%d", session->read_impl.bits_per_second);
|
2008-10-16 16:34:52 +00:00
|
|
|
if (session->read_impl.actual_samples_per_second != session->read_impl.samples_per_second) {
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-reported-read-codec-rate", "%d", session->read_impl.samples_per_second);
|
2008-07-15 17:39:16 +00:00
|
|
|
}
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
2008-10-16 16:34:52 +00:00
|
|
|
switch_channel_set_variable(channel, "read_codec", session->read_impl.iananame);
|
|
|
|
switch_snprintf(tmp, sizeof(tmp), "%d", session->read_impl.actual_samples_per_second);
|
2008-07-15 17:39:16 +00:00
|
|
|
switch_channel_set_variable(channel, "read_rate", tmp);
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2008-07-15 17:39:16 +00:00
|
|
|
session->raw_read_frame.codec = session->read_codec;
|
|
|
|
session->raw_write_frame.codec = session->read_codec;
|
|
|
|
session->enc_read_frame.codec = session->read_codec;
|
|
|
|
session->enc_write_frame.codec = session->read_codec;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
end:
|
2008-10-15 21:54:26 +00:00
|
|
|
|
2010-01-22 00:51:32 +00:00
|
|
|
if (session->read_codec) {
|
|
|
|
switch_channel_set_flag(channel, CF_MEDIA_SET);
|
|
|
|
}
|
|
|
|
|
2008-11-21 00:09:11 +00:00
|
|
|
switch_mutex_unlock(session->codec_read_mutex);
|
2008-10-15 21:54:26 +00:00
|
|
|
return status;
|
2007-06-13 21:51:18 +00:00
|
|
|
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2008-07-09 16:44:19 +00:00
|
|
|
SWITCH_DECLARE(switch_codec_t *) switch_core_session_get_effective_read_codec(switch_core_session_t *session)
|
2007-03-29 22:34:40 +00:00
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
switch_codec_t *codec;
|
|
|
|
codec = session->read_codec;
|
2008-10-15 21:54:26 +00:00
|
|
|
return codec;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2008-07-09 16:44:19 +00:00
|
|
|
SWITCH_DECLARE(switch_codec_t *) switch_core_session_get_read_codec(switch_core_session_t *session)
|
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
switch_codec_t *codec;
|
|
|
|
codec = session->real_read_codec ? session->real_read_codec : session->read_codec;
|
2008-10-15 21:54:26 +00:00
|
|
|
return codec;
|
2008-07-09 16:44:19 +00:00
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_get_read_impl(switch_core_session_t *session, switch_codec_implementation_t *impp)
|
2008-10-16 16:34:52 +00:00
|
|
|
{
|
2010-03-17 16:22:12 +00:00
|
|
|
if (session->read_impl.codec_id) {
|
2008-10-16 16:34:52 +00:00
|
|
|
*impp = session->read_impl;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2015-01-23 17:03:33 +00:00
|
|
|
} else {
|
|
|
|
memset(impp, 0, sizeof(*impp));
|
|
|
|
impp->number_of_channels = 1;
|
2008-10-16 16:34:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2012-05-14 15:03:35 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_get_real_read_impl(switch_core_session_t *session, switch_codec_implementation_t *impp)
|
|
|
|
{
|
|
|
|
if (session->real_read_impl.codec_id) {
|
|
|
|
*impp = session->real_read_impl;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2015-01-23 17:03:33 +00:00
|
|
|
} else {
|
|
|
|
memset(impp, 0, sizeof(*impp));
|
|
|
|
impp->number_of_channels = 1;
|
2012-05-14 15:03:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_get_write_impl(switch_core_session_t *session, switch_codec_implementation_t *impp)
|
2008-10-16 16:34:52 +00:00
|
|
|
{
|
2010-03-17 16:22:12 +00:00
|
|
|
if (session->write_impl.codec_id) {
|
2008-10-16 16:34:52 +00:00
|
|
|
*impp = session->write_impl;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2015-01-23 17:03:33 +00:00
|
|
|
} else {
|
|
|
|
memset(impp, 0, sizeof(*impp));
|
|
|
|
impp->number_of_channels = 1;
|
2008-10-16 16:34:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_get_video_read_impl(switch_core_session_t *session, switch_codec_implementation_t *impp)
|
2008-10-16 16:34:52 +00:00
|
|
|
{
|
2010-03-17 16:22:12 +00:00
|
|
|
if (session->video_read_impl.codec_id) {
|
2008-10-16 16:34:52 +00:00
|
|
|
*impp = session->video_read_impl;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2015-01-23 17:03:33 +00:00
|
|
|
} else {
|
|
|
|
memset(impp, 0, sizeof(*impp));
|
|
|
|
impp->number_of_channels = 1;
|
2008-10-16 16:34:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_get_video_write_impl(switch_core_session_t *session, switch_codec_implementation_t *impp)
|
2008-10-16 16:34:52 +00:00
|
|
|
{
|
2010-03-17 16:22:12 +00:00
|
|
|
if (session->video_write_impl.codec_id) {
|
2008-10-16 16:34:52 +00:00
|
|
|
*impp = session->video_write_impl;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2015-01-23 17:03:33 +00:00
|
|
|
} else {
|
|
|
|
memset(impp, 0, sizeof(*impp));
|
|
|
|
impp->number_of_channels = 1;
|
2008-10-16 16:34:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_read_impl(switch_core_session_t *session, const switch_codec_implementation_t *impp)
|
2009-09-16 04:22:24 +00:00
|
|
|
{
|
2010-02-06 03:38:24 +00:00
|
|
|
session->read_impl = *impp;
|
2009-09-16 04:22:24 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_write_impl(switch_core_session_t *session, const switch_codec_implementation_t *impp)
|
2009-09-16 04:22:24 +00:00
|
|
|
{
|
|
|
|
session->write_impl = *impp;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_video_read_impl(switch_core_session_t *session, const switch_codec_implementation_t *impp)
|
2009-09-16 04:22:24 +00:00
|
|
|
{
|
|
|
|
session->video_read_impl = *impp;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_video_write_impl(switch_core_session_t *session, const switch_codec_implementation_t *impp)
|
2009-09-16 04:22:24 +00:00
|
|
|
{
|
|
|
|
session->video_write_impl = *impp;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-03-30 00:13:31 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_write_codec(switch_core_session_t *session, switch_codec_t *codec)
|
2007-03-29 22:34:40 +00:00
|
|
|
{
|
|
|
|
switch_event_t *event;
|
2008-01-28 07:26:10 +00:00
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
2007-12-04 18:52:13 +00:00
|
|
|
char tmp[30];
|
2008-10-15 21:54:26 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
|
|
|
|
2008-11-21 00:09:11 +00:00
|
|
|
switch_mutex_lock(session->codec_write_mutex);
|
|
|
|
|
2009-04-09 17:17:12 +00:00
|
|
|
if (!codec || !codec->implementation || !switch_core_codec_ready(codec)) {
|
2008-07-09 16:44:19 +00:00
|
|
|
if (session->real_write_codec) {
|
|
|
|
session->write_codec = session->real_write_codec;
|
2008-10-16 16:34:52 +00:00
|
|
|
session->write_impl = *session->real_write_codec->implementation;
|
2008-07-09 16:44:19 +00:00
|
|
|
session->real_write_codec = NULL;
|
|
|
|
} else {
|
2009-08-13 20:35:02 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot set NULL codec!\n");
|
2008-10-15 21:54:26 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
goto end;
|
2008-07-09 16:44:19 +00:00
|
|
|
}
|
|
|
|
} else if (session->write_codec) {
|
|
|
|
if (session->real_write_codec) {
|
|
|
|
if (codec == session->real_write_codec) {
|
|
|
|
session->write_codec = codec;
|
2008-10-16 16:34:52 +00:00
|
|
|
session->write_impl = *codec->implementation;
|
2008-07-09 16:44:19 +00:00
|
|
|
session->real_write_codec = NULL;
|
|
|
|
} else {
|
2009-08-13 20:35:02 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot double-set codec!\n");
|
2008-10-15 21:54:26 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
goto end;
|
2008-07-09 16:44:19 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
session->real_write_codec = session->write_codec;
|
|
|
|
session->write_codec = codec;
|
2008-10-16 16:34:52 +00:00
|
|
|
session->write_impl = *codec->implementation;
|
2008-07-09 16:44:19 +00:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
session->write_codec = codec;
|
2008-10-16 16:34:52 +00:00
|
|
|
session->write_impl = *codec->implementation;
|
2008-07-09 16:44:19 +00:00
|
|
|
}
|
2008-03-27 20:36:03 +00:00
|
|
|
|
2010-03-17 16:22:12 +00:00
|
|
|
if (session->write_codec && codec && session->write_impl.codec_id) {
|
2008-10-16 16:34:52 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_CODEC) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_channel_event_set_data(session->channel, event);
|
2009-10-08 19:37:21 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "Channel-Write-Codec-Name", session->write_impl.iananame);
|
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Channel-Write-Codec-Rate", "%d", session->write_impl.actual_samples_per_second);
|
2010-09-29 21:52:34 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Channel-Write-codec-bit-rate", "%d", session->write_impl.bits_per_second);
|
2008-10-16 16:34:52 +00:00
|
|
|
if (session->write_impl.actual_samples_per_second != session->write_impl.samples_per_second) {
|
2009-10-08 19:37:21 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "Channel-Reported-Write-Codec-Rate", "%d",
|
2008-10-16 16:34:52 +00:00
|
|
|
session->write_impl.actual_samples_per_second);
|
|
|
|
}
|
|
|
|
switch_event_fire(&event);
|
2007-10-19 21:06:09 +00:00
|
|
|
}
|
2007-03-29 22:34:40 +00:00
|
|
|
|
2008-10-16 16:34:52 +00:00
|
|
|
switch_channel_set_variable(channel, "write_codec", session->write_impl.iananame);
|
|
|
|
switch_snprintf(tmp, sizeof(tmp), "%d", session->write_impl.actual_samples_per_second);
|
|
|
|
switch_channel_set_variable(channel, "write_rate", tmp);
|
|
|
|
}
|
2007-12-04 18:52:13 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
end:
|
2008-11-21 00:09:11 +00:00
|
|
|
switch_mutex_unlock(session->codec_write_mutex);
|
2008-10-15 21:54:26 +00:00
|
|
|
|
|
|
|
return status;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2008-07-09 16:44:19 +00:00
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_codec_t *) switch_core_session_get_effective_write_codec(switch_core_session_t *session)
|
2007-03-29 22:34:40 +00:00
|
|
|
{
|
2008-10-15 21:54:26 +00:00
|
|
|
switch_codec_t *codec;
|
|
|
|
codec = session->write_codec;
|
|
|
|
|
|
|
|
return codec;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2008-07-09 16:44:19 +00:00
|
|
|
SWITCH_DECLARE(switch_codec_t *) switch_core_session_get_write_codec(switch_core_session_t *session)
|
|
|
|
{
|
2008-10-15 21:54:26 +00:00
|
|
|
switch_codec_t *codec;
|
|
|
|
codec = session->real_write_codec ? session->real_write_codec : session->write_codec;
|
|
|
|
|
|
|
|
return codec;
|
2008-07-09 16:44:19 +00:00
|
|
|
}
|
|
|
|
|
2008-02-09 00:38:23 +00:00
|
|
|
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_video_read_codec(switch_core_session_t *session, switch_codec_t *codec)
|
|
|
|
{
|
|
|
|
switch_event_t *event;
|
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
char tmp[30];
|
2008-10-15 21:54:26 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
|
|
|
|
2009-04-09 17:17:12 +00:00
|
|
|
if (!codec || !codec->implementation || !switch_core_codec_ready(codec)) {
|
2008-09-30 04:09:25 +00:00
|
|
|
if (session->video_read_codec) {
|
|
|
|
session->video_read_codec = NULL;
|
2010-02-06 03:38:24 +00:00
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
2008-10-15 21:54:26 +00:00
|
|
|
goto end;
|
2008-09-30 04:09:25 +00:00
|
|
|
}
|
2009-08-13 20:35:02 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot set NULL codec!\n");
|
2008-10-15 21:54:26 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
goto end;
|
2008-09-30 04:09:25 +00:00
|
|
|
}
|
|
|
|
|
2008-02-09 00:38:23 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_CODEC) == SWITCH_STATUS_SUCCESS) {
|
|
|
|
switch_channel_event_set_data(session->channel, event);
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "channel-video-read-codec-name", codec->implementation->iananame);
|
2008-02-09 00:38:23 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-video-read-codec-rate", "%d", codec->implementation->actual_samples_per_second);
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_channel_set_variable(channel, "video_read_codec", codec->implementation->iananame);
|
|
|
|
switch_snprintf(tmp, sizeof(tmp), "%d", codec->implementation->actual_samples_per_second);
|
|
|
|
switch_channel_set_variable(channel, "video_read_rate", tmp);
|
|
|
|
|
|
|
|
session->video_read_codec = codec;
|
2010-02-26 19:20:24 +00:00
|
|
|
if (codec->implementation) {
|
|
|
|
session->video_read_impl = *codec->implementation;
|
|
|
|
} else {
|
|
|
|
memset(&session->video_read_impl, 0, sizeof(session->video_read_impl));
|
|
|
|
}
|
2010-02-06 03:38:24 +00:00
|
|
|
end:
|
2008-10-15 21:54:26 +00:00
|
|
|
|
|
|
|
return status;
|
2008-02-09 00:38:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_codec_t *) switch_core_session_get_video_read_codec(switch_core_session_t *session)
|
|
|
|
{
|
2008-10-15 21:54:26 +00:00
|
|
|
switch_codec_t *codec;
|
|
|
|
codec = session->video_read_codec;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2008-10-15 21:54:26 +00:00
|
|
|
return codec;
|
|
|
|
|
2008-02-09 00:38:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_session_set_video_write_codec(switch_core_session_t *session, switch_codec_t *codec)
|
|
|
|
{
|
|
|
|
switch_event_t *event;
|
|
|
|
switch_channel_t *channel = switch_core_session_get_channel(session);
|
|
|
|
char tmp[30];
|
2008-10-15 21:54:26 +00:00
|
|
|
switch_status_t status = SWITCH_STATUS_SUCCESS;
|
2009-04-09 17:17:12 +00:00
|
|
|
if (!codec || !codec->implementation || !switch_core_codec_ready(codec)) {
|
2008-09-30 04:09:25 +00:00
|
|
|
if (session->video_write_codec) {
|
|
|
|
session->video_write_codec = NULL;
|
2010-02-06 03:38:24 +00:00
|
|
|
status = SWITCH_STATUS_SUCCESS;
|
2008-10-15 21:54:26 +00:00
|
|
|
goto end;
|
2008-09-30 04:09:25 +00:00
|
|
|
}
|
2009-08-13 20:35:02 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_SESSION_LOG(session), SWITCH_LOG_ERROR, "Cannot set NULL codec!\n");
|
2008-10-15 21:54:26 +00:00
|
|
|
status = SWITCH_STATUS_FALSE;
|
|
|
|
goto end;
|
2008-09-30 04:09:25 +00:00
|
|
|
}
|
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
if (switch_event_create(&event, SWITCH_EVENT_CODEC) == SWITCH_STATUS_SUCCESS) {
|
2008-02-09 00:38:23 +00:00
|
|
|
switch_channel_event_set_data(session->channel, event);
|
2008-08-16 02:17:09 +00:00
|
|
|
switch_event_add_header_string(event, SWITCH_STACK_BOTTOM, "channel-video-write-codec-name", codec->implementation->iananame);
|
2008-02-09 00:38:23 +00:00
|
|
|
switch_event_add_header(event, SWITCH_STACK_BOTTOM, "channel-video-write-codec-rate", "%d", codec->implementation->actual_samples_per_second);
|
|
|
|
switch_event_fire(&event);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch_channel_set_variable(channel, "video_write_codec", codec->implementation->iananame);
|
|
|
|
switch_snprintf(tmp, sizeof(tmp), "%d", codec->implementation->actual_samples_per_second);
|
|
|
|
switch_channel_set_variable(channel, "video_write_rate", tmp);
|
|
|
|
|
|
|
|
session->video_write_codec = codec;
|
2008-10-16 16:34:52 +00:00
|
|
|
session->video_write_impl = *codec->implementation;
|
2008-10-15 21:54:26 +00:00
|
|
|
|
2010-02-06 03:38:24 +00:00
|
|
|
end:
|
|
|
|
|
2008-10-15 21:54:26 +00:00
|
|
|
return status;
|
2008-02-09 00:38:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_codec_t *) switch_core_session_get_video_write_codec(switch_core_session_t *session)
|
|
|
|
{
|
2008-10-15 21:54:26 +00:00
|
|
|
switch_codec_t *codec;
|
|
|
|
codec = session->video_write_codec;
|
|
|
|
|
|
|
|
return codec;
|
|
|
|
|
2008-02-09 00:38:23 +00:00
|
|
|
}
|
|
|
|
|
2010-10-01 17:22:06 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_codec_parse_fmtp(const char *codec_name, const char *fmtp, uint32_t rate, switch_codec_fmtp_t *codec_fmtp)
|
|
|
|
{
|
|
|
|
switch_codec_interface_t *codec_interface;
|
|
|
|
switch_status_t status = SWITCH_STATUS_FALSE;
|
|
|
|
|
|
|
|
if (zstr(codec_name) || zstr(fmtp) || !codec_fmtp) {
|
|
|
|
return SWITCH_STATUS_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
memset(codec_fmtp, 0, sizeof(*codec_fmtp));
|
|
|
|
|
|
|
|
if ((codec_interface = switch_loadable_module_get_codec_interface(codec_name))) {
|
|
|
|
if (codec_interface->parse_fmtp) {
|
|
|
|
codec_fmtp->actual_samples_per_second = rate;
|
|
|
|
status = codec_interface->parse_fmtp(fmtp, codec_fmtp);
|
|
|
|
}
|
|
|
|
|
|
|
|
UNPROTECT_INTERFACE(codec_interface);
|
|
|
|
}
|
|
|
|
|
|
|
|
return status;
|
|
|
|
}
|
2008-02-09 00:38:23 +00:00
|
|
|
|
2010-05-04 21:03:15 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_codec_reset(switch_codec_t *codec)
|
|
|
|
{
|
|
|
|
switch_assert(codec != NULL);
|
|
|
|
|
|
|
|
codec->implementation->destroy(codec);
|
|
|
|
codec->implementation->init(codec, codec->flags, NULL);
|
|
|
|
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-04-25 18:22:43 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_codec_copy(switch_codec_t *codec, switch_codec_t *new_codec,
|
|
|
|
const switch_codec_settings_t *codec_settings, switch_memory_pool_t *pool)
|
2008-03-17 23:26:38 +00:00
|
|
|
{
|
2015-05-11 21:47:02 +00:00
|
|
|
uint32_t codec_rate;
|
2008-03-17 23:26:38 +00:00
|
|
|
switch_assert(codec != NULL);
|
|
|
|
switch_assert(new_codec != NULL);
|
2015-05-11 21:47:02 +00:00
|
|
|
codec_rate = !strcasecmp(codec->implementation->iananame, "g722")
|
|
|
|
? codec->implementation->samples_per_second : codec->implementation->actual_samples_per_second;
|
|
|
|
|
|
|
|
return switch_core_codec_init(new_codec,
|
2015-02-12 16:14:49 +00:00
|
|
|
codec->implementation->iananame,
|
|
|
|
codec->fmtp_in,
|
2015-05-11 21:47:02 +00:00
|
|
|
codec_rate,
|
2015-02-12 16:14:49 +00:00
|
|
|
codec->implementation->microseconds_per_packet / 1000,
|
|
|
|
codec->implementation->number_of_channels,
|
|
|
|
codec->flags,
|
|
|
|
codec_settings,
|
|
|
|
pool);
|
|
|
|
|
2008-03-17 23:26:38 +00:00
|
|
|
}
|
|
|
|
|
2010-09-29 21:52:34 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_codec_init_with_bitrate(switch_codec_t *codec, const char *codec_name, const char *fmtp,
|
|
|
|
uint32_t rate, int ms, int channels, uint32_t bitrate, uint32_t flags,
|
2007-03-30 00:15:25 +00:00
|
|
|
const switch_codec_settings_t *codec_settings, switch_memory_pool_t *pool)
|
2007-03-29 22:34:40 +00:00
|
|
|
{
|
2008-11-13 17:40:25 +00:00
|
|
|
switch_codec_interface_t *codec_interface;
|
2007-03-29 22:34:40 +00:00
|
|
|
const switch_codec_implementation_t *iptr, *implementation = NULL;
|
|
|
|
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(codec != NULL);
|
|
|
|
switch_assert(codec_name != NULL);
|
2007-03-29 22:34:40 +00:00
|
|
|
|
|
|
|
memset(codec, 0, sizeof(*codec));
|
|
|
|
|
|
|
|
if ((codec_interface = switch_loadable_module_get_codec_interface(codec_name)) == 0) {
|
2008-10-11 05:44:11 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Invalid codec %s!\n", codec_name);
|
2007-03-29 22:34:40 +00:00
|
|
|
return SWITCH_STATUS_GENERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* If no specific codec interval is requested opt for 20ms above all else because lots of stuff assumes it */
|
|
|
|
if (!ms) {
|
|
|
|
for (iptr = codec_interface->implementations; iptr; iptr = iptr->next) {
|
2013-10-24 00:04:23 +00:00
|
|
|
uint32_t crate = !strcasecmp(codec_name, "g722") ? iptr->samples_per_second : iptr->actual_samples_per_second;
|
|
|
|
if ((!rate || rate == crate) && (!bitrate || bitrate == (uint32_t)iptr->bits_per_second) &&
|
2008-10-20 17:48:42 +00:00
|
|
|
(20 == (iptr->microseconds_per_packet / 1000)) && (!channels || channels == iptr->number_of_channels)) {
|
2007-03-29 22:34:40 +00:00
|
|
|
implementation = iptr;
|
|
|
|
goto found;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Either looking for a specific interval or there was no interval specified and there wasn't one @20ms available */
|
|
|
|
for (iptr = codec_interface->implementations; iptr; iptr = iptr->next) {
|
2013-10-24 00:04:23 +00:00
|
|
|
uint32_t crate = !strcasecmp(codec_name, "g722") ? iptr->samples_per_second : iptr->actual_samples_per_second;
|
|
|
|
if ((!rate || rate == crate) && (!bitrate || bitrate == (uint32_t)iptr->bits_per_second) &&
|
2008-10-20 17:48:42 +00:00
|
|
|
(!ms || ms == (iptr->microseconds_per_packet / 1000)) && (!channels || channels == iptr->number_of_channels)) {
|
2007-03-29 22:34:40 +00:00
|
|
|
implementation = iptr;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
found:
|
|
|
|
|
|
|
|
if (implementation) {
|
|
|
|
switch_status_t status;
|
|
|
|
codec->codec_interface = codec_interface;
|
|
|
|
codec->implementation = implementation;
|
|
|
|
codec->flags = flags;
|
|
|
|
|
|
|
|
if (pool) {
|
|
|
|
codec->memory_pool = pool;
|
|
|
|
} else {
|
|
|
|
if ((status = switch_core_new_memory_pool(&codec->memory_pool)) != SWITCH_STATUS_SUCCESS) {
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
switch_set_flag(codec, SWITCH_CODEC_FLAG_FREE_POOL);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fmtp) {
|
|
|
|
codec->fmtp_in = switch_core_strdup(codec->memory_pool, fmtp);
|
|
|
|
}
|
|
|
|
|
|
|
|
implementation->init(codec, flags, codec_settings);
|
2008-11-12 15:34:34 +00:00
|
|
|
switch_mutex_init(&codec->mutex, SWITCH_MUTEX_NESTED, codec->memory_pool);
|
2009-04-09 17:17:12 +00:00
|
|
|
switch_set_flag(codec, SWITCH_CODEC_FLAG_READY);
|
2007-03-29 22:34:40 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
} else {
|
2014-06-12 17:06:33 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Codec %s Exists but not at the desired implementation. %dhz %dms %dch\n",
|
|
|
|
codec_name, rate, ms, channels);
|
|
|
|
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2008-11-12 19:28:05 +00:00
|
|
|
UNPROTECT_INTERFACE(codec_interface);
|
|
|
|
|
2007-03-29 22:34:40 +00:00
|
|
|
return SWITCH_STATUS_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_codec_encode(switch_codec_t *codec,
|
|
|
|
switch_codec_t *other_codec,
|
|
|
|
void *decoded_data,
|
|
|
|
uint32_t decoded_data_len,
|
|
|
|
uint32_t decoded_rate,
|
2008-05-27 04:30:03 +00:00
|
|
|
void *encoded_data, uint32_t *encoded_data_len, uint32_t *encoded_rate, unsigned int *flag)
|
2007-03-29 22:34:40 +00:00
|
|
|
{
|
2008-11-12 15:34:34 +00:00
|
|
|
switch_status_t status;
|
|
|
|
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(codec != NULL);
|
|
|
|
switch_assert(encoded_data != NULL);
|
|
|
|
switch_assert(decoded_data != NULL);
|
2007-03-29 22:34:40 +00:00
|
|
|
|
2009-04-09 17:17:12 +00:00
|
|
|
if (!codec->implementation || !switch_core_codec_ready(codec)) {
|
2008-01-07 11:21:24 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Codec is not initialized!\n");
|
2008-10-29 00:04:20 +00:00
|
|
|
return SWITCH_STATUS_NOT_INITALIZED;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!switch_test_flag(codec, SWITCH_CODEC_FLAG_ENCODE)) {
|
2008-09-01 04:43:57 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Codec encoder is not initialized!\n");
|
2008-10-29 00:04:20 +00:00
|
|
|
return SWITCH_STATUS_NOT_INITALIZED;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2010-07-06 18:20:27 +00:00
|
|
|
if (codec->mutex) switch_mutex_lock(codec->mutex);
|
2010-02-06 03:38:24 +00:00
|
|
|
status = codec->implementation->encode(codec, other_codec, decoded_data, decoded_data_len,
|
2008-11-12 15:34:34 +00:00
|
|
|
decoded_rate, encoded_data, encoded_data_len, encoded_rate, flag);
|
2010-07-06 18:20:27 +00:00
|
|
|
if (codec->mutex) switch_mutex_unlock(codec->mutex);
|
2008-11-12 15:34:34 +00:00
|
|
|
|
|
|
|
return status;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_codec_decode(switch_codec_t *codec,
|
|
|
|
switch_codec_t *other_codec,
|
|
|
|
void *encoded_data,
|
|
|
|
uint32_t encoded_data_len,
|
|
|
|
uint32_t encoded_rate,
|
2008-05-27 04:30:03 +00:00
|
|
|
void *decoded_data, uint32_t *decoded_data_len, uint32_t *decoded_rate, unsigned int *flag)
|
2007-03-29 22:34:40 +00:00
|
|
|
{
|
2008-11-12 15:34:34 +00:00
|
|
|
switch_status_t status;
|
|
|
|
|
2007-12-11 19:23:57 +00:00
|
|
|
switch_assert(codec != NULL);
|
|
|
|
switch_assert(encoded_data != NULL);
|
|
|
|
switch_assert(decoded_data != NULL);
|
2007-03-29 22:34:40 +00:00
|
|
|
|
2009-04-09 17:17:12 +00:00
|
|
|
if (!codec->implementation || !switch_core_codec_ready(codec)) {
|
2012-12-09 20:01:28 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Decode Codec is not initialized!\n");
|
2008-10-29 00:04:20 +00:00
|
|
|
return SWITCH_STATUS_NOT_INITALIZED;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!switch_test_flag(codec, SWITCH_CODEC_FLAG_DECODE)) {
|
2008-09-01 04:43:57 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "Codec decoder is not initialized!\n");
|
2008-10-29 00:04:20 +00:00
|
|
|
return SWITCH_STATUS_NOT_INITALIZED;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2010-03-25 21:36:11 +00:00
|
|
|
if (codec->implementation->encoded_bytes_per_packet) {
|
2014-06-12 17:06:33 +00:00
|
|
|
uint32_t frames = encoded_data_len / codec->implementation->encoded_bytes_per_packet / codec->implementation->number_of_channels;
|
2010-03-25 21:36:11 +00:00
|
|
|
|
|
|
|
if (frames && codec->implementation->decoded_bytes_per_packet * frames > *decoded_data_len) {
|
2012-02-10 23:27:50 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_CRIT, "Buffer size sanity check failed! edl:%u ebpp:%u fr:%u ddl:%u\n",
|
|
|
|
encoded_data_len, codec->implementation->encoded_bytes_per_packet, frames, *decoded_data_len);
|
2011-04-19 18:51:08 +00:00
|
|
|
*decoded_data_len = codec->implementation->decoded_bytes_per_packet;
|
|
|
|
memset(decoded_data, 255, *decoded_data_len);
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
2010-03-25 21:36:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-07-06 18:20:27 +00:00
|
|
|
if (codec->mutex) switch_mutex_lock(codec->mutex);
|
2010-02-06 03:38:24 +00:00
|
|
|
status = codec->implementation->decode(codec, other_codec, encoded_data, encoded_data_len, encoded_rate,
|
2008-11-12 15:34:34 +00:00
|
|
|
decoded_data, decoded_data_len, decoded_rate, flag);
|
2010-07-06 18:20:27 +00:00
|
|
|
if (codec->mutex) switch_mutex_unlock(codec->mutex);
|
2008-11-12 15:34:34 +00:00
|
|
|
|
|
|
|
return status;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(switch_status_t) switch_core_codec_destroy(switch_codec_t *codec)
|
|
|
|
{
|
2010-11-19 17:16:18 +00:00
|
|
|
switch_mutex_t *mutex = codec->mutex;
|
|
|
|
switch_memory_pool_t *pool = codec->memory_pool;
|
2008-12-18 23:07:32 +00:00
|
|
|
int free_pool = 0;
|
2010-02-06 03:38:24 +00:00
|
|
|
|
|
|
|
switch_assert(codec != NULL);
|
|
|
|
|
2010-11-19 17:16:18 +00:00
|
|
|
if (mutex) switch_mutex_lock(mutex);
|
|
|
|
|
|
|
|
if (switch_core_codec_ready(codec)) {
|
|
|
|
switch_clear_flag(codec, SWITCH_CODEC_FLAG_READY);
|
|
|
|
} else {
|
2008-01-07 11:21:24 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_WARNING, "Codec is not initialized!\n");
|
2010-11-19 17:16:18 +00:00
|
|
|
if (mutex) switch_mutex_unlock(mutex);
|
2008-10-29 00:04:20 +00:00
|
|
|
return SWITCH_STATUS_NOT_INITALIZED;
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2008-12-18 23:07:32 +00:00
|
|
|
if (switch_test_flag(codec, SWITCH_CODEC_FLAG_FREE_POOL)) {
|
|
|
|
free_pool = 1;
|
|
|
|
}
|
|
|
|
|
2007-03-29 22:34:40 +00:00
|
|
|
codec->implementation->destroy(codec);
|
2010-07-06 18:20:27 +00:00
|
|
|
|
2008-11-12 19:28:05 +00:00
|
|
|
UNPROTECT_INTERFACE(codec->codec_interface);
|
|
|
|
|
2010-11-19 17:16:18 +00:00
|
|
|
if (mutex) switch_mutex_unlock(mutex);
|
2010-07-06 18:20:27 +00:00
|
|
|
|
2008-12-18 23:07:32 +00:00
|
|
|
if (free_pool) {
|
2008-11-12 15:34:34 +00:00
|
|
|
switch_core_destroy_memory_pool(&pool);
|
2007-03-29 22:34:40 +00:00
|
|
|
}
|
|
|
|
|
2010-07-06 18:20:27 +00:00
|
|
|
memset(codec, 0, sizeof(*codec));
|
|
|
|
|
2007-03-29 22:34:40 +00:00
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
2008-01-27 17:36:53 +00:00
|
|
|
|
|
|
|
/* For Emacs:
|
|
|
|
* Local Variables:
|
|
|
|
* mode:c
|
2008-02-03 22:14:57 +00:00
|
|
|
* indent-tabs-mode:t
|
2008-01-27 17:36:53 +00:00
|
|
|
* tab-width:4
|
|
|
|
* c-basic-offset:4
|
|
|
|
* End:
|
|
|
|
* For VIM:
|
2013-06-25 16:50:17 +00:00
|
|
|
* vim:set softtabstop=4 shiftwidth=4 tabstop=4 noet:
|
2008-01-27 17:36:53 +00:00
|
|
|
*/
|