mirror of
https://github.com/signalwire/freeswitch.git
synced 2025-06-01 03:05:50 +00:00
git-svn-id: http://svn.freeswitch.org/svn/freeswitch/trunk@247 d0543943-73ff-0310-b7d9-9358b9ac24b2
401 lines
12 KiB
C
401 lines
12 KiB
C
/*
|
|
* 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>
|
|
*
|
|
*
|
|
* mod_rawaudio.c -- Raw Signed Linear Codec
|
|
*
|
|
*/
|
|
#include <switch.h>
|
|
#include <libresample.h>
|
|
|
|
static const char modname[] = "mod_rawaudio";
|
|
|
|
#ifndef MIN
|
|
#define MIN(a,b) ((a) < (b) ? (a) : (b))
|
|
#endif
|
|
|
|
#ifndef MAX
|
|
#define MAX(a,b) ((a) > (b) ? (a) : (b))
|
|
#endif
|
|
|
|
#define QUALITY 1
|
|
|
|
struct raw_resampler {
|
|
void *resampler;
|
|
int from;
|
|
int to;
|
|
double factor;
|
|
float *buf;
|
|
int buf_len;
|
|
int buf_size;
|
|
float *new_buf;
|
|
int new_buf_len;
|
|
int new_buf_size;
|
|
};
|
|
|
|
struct raw_context {
|
|
struct raw_resampler *enc;
|
|
struct raw_resampler *dec;
|
|
};
|
|
|
|
|
|
static int resample(void *handle, double factor, float *src, int srclen, float *dst, int dstlen, int last)
|
|
{
|
|
int o=0, srcused=0, srcpos=0, out=0;
|
|
|
|
for(;;) {
|
|
int srcBlock = MIN(srclen-srcpos, srclen);
|
|
int lastFlag = (last && (srcBlock == srclen-srcpos));
|
|
o = resample_process(handle, factor, &src[srcpos], srcBlock, lastFlag, &srcused, &dst[out], dstlen-out);
|
|
//printf("resampling %d/%d (%d) %d %f\n", srcpos, srclen, MIN(dstlen-out, dstlen), srcused, factor);
|
|
|
|
srcpos += srcused;
|
|
if (o >= 0) {
|
|
out += o;
|
|
}
|
|
if (o < 0 || (o == 0 && srcpos == srclen)) {
|
|
break;
|
|
}
|
|
}
|
|
return out;
|
|
}
|
|
|
|
|
|
static switch_status switch_raw_init(switch_codec *codec, switch_codec_flag flags, const struct switch_codec_settings *codec_settings)
|
|
{
|
|
int encoding, decoding;
|
|
struct raw_context *context = NULL;
|
|
|
|
encoding = (flags & SWITCH_CODEC_FLAG_ENCODE);
|
|
decoding = (flags & SWITCH_CODEC_FLAG_DECODE);
|
|
|
|
if (!(encoding || decoding)) {
|
|
return SWITCH_STATUS_FALSE;
|
|
} else {
|
|
if (!(context = switch_core_alloc(codec->memory_pool, sizeof(*context)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
codec->private = context;
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
}
|
|
|
|
static switch_status switch_raw_encode(switch_codec *codec,
|
|
switch_codec *other_codec,
|
|
void *decoded_data,
|
|
size_t decoded_data_len,
|
|
int decoded_rate,
|
|
void *encoded_data,
|
|
size_t *encoded_data_len,
|
|
int *encoded_rate,
|
|
unsigned int *flag)
|
|
{
|
|
struct raw_context *context = codec->private;
|
|
|
|
/* NOOP indicates that the audio in is already the same as the audio out, so no conversion was necessary.
|
|
TBD Support varying number of channels
|
|
*/
|
|
//printf("encode %d->%d (%d)\n", other_codec->implementation->samples_per_second, codec->implementation->samples_per_second, decoded_rate);
|
|
|
|
if (other_codec &&
|
|
codec->implementation->samples_per_second != other_codec->implementation->samples_per_second &&
|
|
decoded_rate != other_codec->implementation->samples_per_second) {
|
|
const short *ddp = decoded_data;
|
|
short *edp = encoded_data;
|
|
size_t ddplen = decoded_data_len / 2;
|
|
|
|
if (!context->enc) {
|
|
|
|
if (!(context->enc = switch_core_alloc(codec->memory_pool, sizeof(struct raw_resampler)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
|
|
context->enc->from = other_codec->implementation->samples_per_second;
|
|
context->enc->to = codec->implementation->samples_per_second;
|
|
context->enc->factor = ((double)context->enc->to / (double)context->enc->from);
|
|
|
|
context->enc->resampler = resample_open(QUALITY, context->enc->factor, context->enc->factor);
|
|
switch_console_printf(SWITCH_CHANNEL_CONSOLE, "Activate Encode Resample %d->%d %f\n", context->enc->from, context->enc->to, context->enc->factor);
|
|
context->enc->buf_size = codec->implementation->bytes_per_frame * 10;
|
|
context->enc->buf = (float *) switch_core_alloc(codec->memory_pool, context->enc->buf_size);
|
|
context->enc->new_buf_size = codec->implementation->bytes_per_frame * 10;
|
|
context->enc->new_buf = (float *) switch_core_alloc(codec->memory_pool, context->enc->new_buf_size);
|
|
}
|
|
|
|
if (context->enc) {
|
|
context->enc->buf_len = switch_short_to_float(decoded_data, context->enc->buf, (int)ddplen);
|
|
context->enc->new_buf_len = resample(context->enc->resampler,
|
|
context->enc->factor,
|
|
context->enc->buf,
|
|
context->enc->buf_len,
|
|
context->enc->new_buf,
|
|
context->enc->new_buf_size,
|
|
0);
|
|
switch_float_to_short(context->enc->new_buf, edp, decoded_data_len * 2);
|
|
*encoded_data_len = context->enc->new_buf_len * 2;
|
|
*encoded_rate = context->enc->to;
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
}
|
|
|
|
return SWITCH_STATUS_NOOP;
|
|
}
|
|
|
|
static switch_status switch_raw_decode(switch_codec *codec,
|
|
switch_codec *other_codec,
|
|
void *encoded_data,
|
|
size_t encoded_data_len,
|
|
int encoded_rate,
|
|
void *decoded_data,
|
|
size_t *decoded_data_len,
|
|
int *decoded_rate,
|
|
unsigned int *flag)
|
|
{
|
|
struct raw_context *context = codec->private;
|
|
|
|
//printf("decode %d->%d (%d)\n", other_codec->implementation->samples_per_second, codec->implementation->samples_per_second, encoded_rate);
|
|
|
|
|
|
if (other_codec &&
|
|
codec->implementation->samples_per_second != other_codec->implementation->samples_per_second &&
|
|
encoded_rate != other_codec->implementation->samples_per_second) {
|
|
short *ddp = decoded_data;
|
|
const short *edp = encoded_data;
|
|
size_t edplen = encoded_data_len / 2;
|
|
|
|
if (!context->dec) {
|
|
if (!(context->dec = switch_core_alloc(codec->memory_pool, sizeof(struct raw_resampler)))) {
|
|
return SWITCH_STATUS_MEMERR;
|
|
}
|
|
|
|
|
|
context->dec->from = other_codec->implementation->samples_per_second;
|
|
context->dec->to = codec->implementation->samples_per_second;
|
|
context->dec->factor = ((double)context->dec->from / (double)context->dec->to);
|
|
|
|
context->dec->resampler = resample_open(QUALITY, context->dec->factor, context->dec->factor);
|
|
switch_console_printf(SWITCH_CHANNEL_CONSOLE, "Activate Decode Resample %d->%d %f\n", context->dec->from, context->dec->to, context->dec->factor);
|
|
|
|
context->dec->buf_size = codec->implementation->bytes_per_frame * 10;
|
|
context->dec->buf = (float *) switch_core_alloc(codec->memory_pool, context->dec->buf_size);
|
|
context->dec->new_buf_size = codec->implementation->bytes_per_frame * 10;
|
|
context->dec->new_buf = (float *) switch_core_alloc(codec->memory_pool, context->dec->new_buf_size);
|
|
}
|
|
|
|
if (context->dec) {
|
|
context->dec->buf_len = switch_short_to_float(encoded_data, context->dec->buf, (int)edplen);
|
|
context->dec->new_buf_len = resample(context->dec->resampler,
|
|
context->dec->factor,
|
|
context->dec->buf,
|
|
context->dec->buf_len,
|
|
context->dec->new_buf,
|
|
context->dec->new_buf_size,
|
|
0);
|
|
switch_float_to_short(context->dec->new_buf, ddp, (int)edplen);
|
|
*decoded_data_len = context->dec->new_buf_len * 2;
|
|
*decoded_rate = context->dec->to;
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
return SWITCH_STATUS_GENERR;
|
|
}
|
|
|
|
|
|
return SWITCH_STATUS_NOOP;
|
|
}
|
|
|
|
|
|
static switch_status switch_raw_destroy(switch_codec *codec)
|
|
{
|
|
struct raw_context *context = codec->private;
|
|
|
|
if (context->enc && context->enc->resampler){
|
|
resample_close(context->enc->resampler);
|
|
}
|
|
|
|
if (context->dec && context->dec->resampler){
|
|
resample_close(context->dec->resampler);
|
|
}
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
#if 0
|
|
switch_status raw_file_open(switch_file_handle *handle, char *path)
|
|
{
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
switch_status raw_file_close(switch_file_handle *handle)
|
|
{
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
switch_status raw_file_seek(switch_file_handle *handle, unsigned int *cur_sample, unsigned int samples, int whence)
|
|
{
|
|
return SWITCH_STATUS_NOTIMPL;
|
|
}
|
|
|
|
|
|
/* Registration */
|
|
|
|
|
|
static char *supported_formats[] = {"raw", "r8k", NULL};
|
|
|
|
static const switch_file_interface raw_file_interface = {
|
|
/*.interface_name*/ "raw",
|
|
/*.file_open*/ raw_file_open,
|
|
/*.file_close*/ raw_file_close,
|
|
/*.file_read*/ NULL,
|
|
/*.file_write*/ NULL,
|
|
/*.file_seek*/ raw_file_seek,
|
|
/*.extens*/ supported_formats,
|
|
/*.next*/ NULL,
|
|
};
|
|
#endif
|
|
|
|
static const switch_codec_implementation raw_32k_implementation = {
|
|
/*.samples_per_second = */ 32000,
|
|
/*.bits_per_second = */ 512000,
|
|
/*.microseconds_per_frame = */ 20000,
|
|
/*.samples_per_frame = */ 640,
|
|
/*.bytes_per_frame = */ 1280,
|
|
/*.encoded_bytes_per_frame = */ 1280,
|
|
/*.number_of_channels = */ 1,
|
|
/*.pref_frames_per_packet = */ 1,
|
|
/*.max_frames_per_packet = */ 1,
|
|
/*.init = */ switch_raw_init,
|
|
/*.encode = */ switch_raw_encode,
|
|
/*.decode = */ switch_raw_decode,
|
|
/*.destroy = */ switch_raw_destroy
|
|
};
|
|
|
|
static const switch_codec_implementation raw_22k_implementation = {
|
|
/*.samples_per_second = */ 22050,
|
|
/*.bits_per_second = */ 352800,
|
|
/*.microseconds_per_frame = */ 20000,
|
|
/*.samples_per_frame = */ 441,
|
|
/*.bytes_per_frame = */ 882,
|
|
/*.encoded_bytes_per_frame = */ 882,
|
|
/*.number_of_channels = */ 1,
|
|
/*.pref_frames_per_packet = */ 1,
|
|
/*.max_frames_per_packet = */ 1,
|
|
/*.init = */ switch_raw_init,
|
|
/*.encode = */ switch_raw_encode,
|
|
/*.decode = */ switch_raw_decode,
|
|
/*.destroy = */ switch_raw_destroy,
|
|
/*.next = */ &raw_32k_implementation
|
|
};
|
|
|
|
static const switch_codec_implementation raw_16k_implementation = {
|
|
/*.samples_per_second = */ 16000,
|
|
/*.bits_per_second = */ 256000,
|
|
/*.microseconds_per_frame = */ 20000,
|
|
/*.samples_per_frame = */ 320,
|
|
/*.bytes_per_frame = */ 640,
|
|
/*.encoded_bytes_per_frame = */ 640,
|
|
/*.number_of_channels = */ 1,
|
|
/*.pref_frames_per_packet = */ 1,
|
|
/*.max_frames_per_packet = */ 1,
|
|
/*.init = */ switch_raw_init,
|
|
/*.encode = */ switch_raw_encode,
|
|
/*.decode = */ switch_raw_decode,
|
|
/*.destroy = */ switch_raw_destroy,
|
|
/*.next = */ &raw_22k_implementation
|
|
};
|
|
|
|
static const switch_codec_implementation raw_8k_implementation = {
|
|
/*.samples_per_second = */ 8000,
|
|
/*.bits_per_second = */ 128000,
|
|
/*.microseconds_per_frame = */ 20000,
|
|
/*.samples_per_frame = */ 160,
|
|
/*.bytes_per_frame = */ 320,
|
|
/*.encoded_bytes_per_frame = */ 320,
|
|
/*.number_of_channels = */ 1,
|
|
/*.pref_frames_per_packet = */ 1,
|
|
/*.max_frames_per_packet = */ 1,
|
|
/*.init = */ switch_raw_init,
|
|
/*.encode = */ switch_raw_encode,
|
|
/*.decode = */ switch_raw_decode,
|
|
/*.destroy = */ switch_raw_destroy,
|
|
/*.next = */ &raw_16k_implementation
|
|
};
|
|
|
|
|
|
static const switch_codec_implementation raw_8k_30ms_implementation = {
|
|
/*.samples_per_second*/ 8000,
|
|
/*.bits_per_second*/ 128000,
|
|
/*.microseconds_per_frame*/ 30000,
|
|
/*.samples_per_frame*/ 240,
|
|
/*.bytes_per_frame*/ 480,
|
|
/*.encoded_bytes_per_frame*/ 480,
|
|
/*.number_of_channels*/ 1,
|
|
/*.pref_frames_per_packet*/ 1,
|
|
/*.max_frames_per_packet*/ 1,
|
|
/*.init*/ switch_raw_init,
|
|
/*.encode*/ switch_raw_encode,
|
|
/*.decode*/ switch_raw_decode,
|
|
/*.destroy*/ switch_raw_destroy,
|
|
/*.next*/ &raw_8k_implementation
|
|
};
|
|
|
|
|
|
static const switch_codec_interface raw_codec_interface = {
|
|
/*.interface_name*/ "raw signed linear (16 bit)",
|
|
/*.codec_type*/ SWITCH_CODEC_TYPE_AUDIO,
|
|
/*.ianacode*/ 10,
|
|
/*.iananame*/ "L16",
|
|
/*.implementations*/ &raw_8k_30ms_implementation
|
|
};
|
|
|
|
static switch_loadable_module_interface raw_module_interface = {
|
|
/*.module_name*/ modname,
|
|
/*.endpoint_interface*/ NULL,
|
|
/*.timer_interface*/ NULL,
|
|
/*.dialplan_interface*/ NULL,
|
|
/*.codec_interface*/ &raw_codec_interface,
|
|
/*.application_interface*/ NULL,
|
|
/*.api_interface*/ NULL,
|
|
///*.file_interface*/ &raw_file_interface
|
|
};
|
|
|
|
|
|
SWITCH_MOD_DECLARE(switch_status) switch_module_load(const switch_loadable_module_interface **interface, char *filename) {
|
|
/* connect my internal structure to the blank pointer passed to me */
|
|
*interface = &raw_module_interface;
|
|
|
|
/* indicate that the module should continue to be loaded */
|
|
return SWITCH_STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
|
|
|
|
|