2006-01-02 17:28:59 +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_caller.c -- Caller Identification
|
|
|
|
*
|
|
|
|
*/
|
2006-04-29 01:00:52 +00:00
|
|
|
#include <switch.h>
|
2006-04-29 06:05:03 +00:00
|
|
|
#include <switch_resample.h>
|
2006-01-02 17:28:59 +00:00
|
|
|
#include <libresample.h>
|
|
|
|
#define NORMFACT (float)0x8000
|
|
|
|
#define MAXSAMPLE (float)0x7FFF
|
|
|
|
#define MAXSAMPLEC (char)0x7F
|
|
|
|
#define QUALITY 1
|
|
|
|
|
|
|
|
#ifndef MIN
|
|
|
|
#define MIN(a,b) ((a) < (b) ? (a) : (b))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef MAX
|
|
|
|
#define MAX(a,b) ((a) > (b) ? (a) : (b))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
2006-04-29 23:43:28 +00:00
|
|
|
SWITCH_DECLARE(switch_status_t) switch_resample_create(switch_audio_resampler_t **new_resampler,
|
2006-01-02 17:28:59 +00:00
|
|
|
int from_rate,
|
2006-03-30 23:02:50 +00:00
|
|
|
switch_size_t from_size,
|
2006-04-29 01:00:52 +00:00
|
|
|
int to_rate, uint32_t to_size, switch_memory_pool_t *pool)
|
2006-01-02 17:28:59 +00:00
|
|
|
{
|
2006-04-29 01:00:52 +00:00
|
|
|
switch_audio_resampler_t *resampler;
|
2006-03-30 23:02:50 +00:00
|
|
|
double lto_rate, lfrom_rate;
|
2006-01-03 01:17:59 +00:00
|
|
|
|
2006-02-20 00:23:25 +00:00
|
|
|
if ((resampler = switch_core_alloc(pool, sizeof(*resampler))) == 0) {
|
2006-01-02 17:28:59 +00:00
|
|
|
return SWITCH_STATUS_MEMERR;
|
|
|
|
}
|
|
|
|
|
|
|
|
resampler->from_rate = from_rate;
|
|
|
|
resampler->to_rate = to_rate;
|
2006-03-30 23:02:50 +00:00
|
|
|
lto_rate = (double) resampler->to_rate;
|
|
|
|
lfrom_rate = (double) resampler->from_rate;
|
|
|
|
resampler->factor = (lto_rate / lfrom_rate);
|
2006-01-02 17:28:59 +00:00
|
|
|
|
|
|
|
resampler->resampler = resample_open(QUALITY, resampler->factor, resampler->factor);
|
2006-04-16 06:05:53 +00:00
|
|
|
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_NOTICE, "Activate Resampler %d->%d %f\n", resampler->from_rate,
|
2006-01-20 15:05:05 +00:00
|
|
|
resampler->to_rate, resampler->factor);
|
2006-01-02 17:28:59 +00:00
|
|
|
resampler->from_size = from_size;
|
|
|
|
resampler->from = (float *) switch_core_alloc(pool, resampler->from_size);
|
|
|
|
resampler->to_size = to_size;
|
|
|
|
resampler->to = (float *) switch_core_alloc(pool, resampler->to_size);
|
|
|
|
|
|
|
|
*new_resampler = resampler;
|
|
|
|
return SWITCH_STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-04-29 01:00:52 +00:00
|
|
|
SWITCH_DECLARE(uint32_t) switch_resample_process(switch_audio_resampler_t *resampler, float *src, int srclen, float *dst,
|
2006-04-09 00:10:13 +00:00
|
|
|
uint32_t dstlen, int last)
|
2006-01-02 17:28:59 +00:00
|
|
|
{
|
2006-01-20 15:05:05 +00:00
|
|
|
int o = 0, srcused = 0, srcpos = 0, out = 0;
|
2006-01-02 17:28:59 +00:00
|
|
|
|
2006-01-20 15:05:05 +00:00
|
|
|
for (;;) {
|
|
|
|
int srcBlock = MIN(srclen - srcpos, srclen);
|
|
|
|
int lastFlag = (last && (srcBlock == srclen - srcpos));
|
|
|
|
o = resample_process(resampler->resampler, resampler->factor, &src[srcpos], srcBlock, lastFlag, &srcused,
|
|
|
|
&dst[out], dstlen - out);
|
2006-01-03 01:17:59 +00:00
|
|
|
//printf("resampling %d/%d (%d) %d %f\n", srcpos, srclen, MIN(dstlen-out, dstlen), srcused, factor);
|
2006-01-02 17:28:59 +00:00
|
|
|
|
|
|
|
srcpos += srcused;
|
|
|
|
if (o >= 0) {
|
2006-01-03 01:17:59 +00:00
|
|
|
out += o;
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
|
|
|
if (o < 0 || (o == 0 && srcpos == srclen)) {
|
2006-01-03 01:17:59 +00:00
|
|
|
break;
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
2006-01-03 01:17:59 +00:00
|
|
|
}
|
|
|
|
return out;
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
|
|
|
|
2006-04-29 01:00:52 +00:00
|
|
|
SWITCH_DECLARE(void) switch_resample_destroy(switch_audio_resampler_t *resampler)
|
2006-01-02 17:28:59 +00:00
|
|
|
{
|
|
|
|
resample_close(resampler->resampler);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2006-03-30 23:02:50 +00:00
|
|
|
SWITCH_DECLARE(switch_size_t) switch_float_to_short(float *f, short *s, switch_size_t len)
|
2006-01-02 17:28:59 +00:00
|
|
|
{
|
2006-03-30 23:02:50 +00:00
|
|
|
switch_size_t i;
|
2006-01-02 17:28:59 +00:00
|
|
|
float ft;
|
2006-01-20 15:05:05 +00:00
|
|
|
for (i = 0; i < len; i++) {
|
2006-01-02 17:28:59 +00:00
|
|
|
ft = f[i] * NORMFACT;
|
2006-01-20 15:05:05 +00:00
|
|
|
if (ft >= 0) {
|
|
|
|
s[i] = (short) (ft + 0.5);
|
2006-01-02 17:28:59 +00:00
|
|
|
} else {
|
2006-01-20 15:05:05 +00:00
|
|
|
s[i] = (short) (ft - 0.5);
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
2006-01-20 15:05:05 +00:00
|
|
|
if ((float) s[i] > MAXSAMPLE)
|
|
|
|
s[i] = (short) MAXSAMPLE;
|
|
|
|
if (s[i] < (short) -MAXSAMPLE)
|
|
|
|
s[i] = (short) -MAXSAMPLE;
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_char_to_float(char *c, float *f, int len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (len % 2) {
|
2006-01-20 15:05:05 +00:00
|
|
|
return (-1);
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
|
|
|
|
2006-01-20 15:05:05 +00:00
|
|
|
for (i = 1; i < len; i += 2) {
|
|
|
|
f[(int) (i / 2)] = (float) (((c[i]) * 0x100) + c[i - 1]);
|
|
|
|
f[(int) (i / 2)] /= NORMFACT;
|
|
|
|
if (f[(int) (i / 2)] > MAXSAMPLE)
|
|
|
|
f[(int) (i / 2)] = MAXSAMPLE;
|
|
|
|
if (f[(int) (i / 2)] < -MAXSAMPLE)
|
|
|
|
f[(int) (i / 2)] = -MAXSAMPLE;
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
2006-01-20 15:05:05 +00:00
|
|
|
return len / 2;
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_float_to_char(float *f, char *c, int len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
float ft;
|
|
|
|
long l;
|
2006-01-20 15:05:05 +00:00
|
|
|
for (i = 0; i < len; i++) {
|
2006-01-02 17:28:59 +00:00
|
|
|
ft = f[i] * NORMFACT;
|
|
|
|
if (ft >= 0) {
|
2006-01-20 15:05:05 +00:00
|
|
|
l = (long) (ft + 0.5);
|
2006-01-02 17:28:59 +00:00
|
|
|
} else {
|
2006-01-20 15:05:05 +00:00
|
|
|
l = (long) (ft - 0.5);
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
2006-01-20 15:05:05 +00:00
|
|
|
c[i * 2] = (unsigned char) ((l) & 0xff);
|
|
|
|
c[i * 2 + 1] = (unsigned char) (((l) >> 8) & 0xff);
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
2006-01-20 15:05:05 +00:00
|
|
|
return len * 2;
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
SWITCH_DECLARE(int) switch_short_to_float(short *s, float *f, int len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
2006-01-20 15:05:05 +00:00
|
|
|
for (i = 0; i < len; i++) {
|
|
|
|
f[i] = (float) (s[i]) / NORMFACT;
|
2006-01-02 17:28:59 +00:00
|
|
|
//f[i] = (float) s[i];
|
|
|
|
}
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
SWITCH_DECLARE(void) switch_swap_linear(int16_t *buf, int len)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < len; i++) {
|
2006-01-20 15:05:05 +00:00
|
|
|
buf[i] = ((buf[i] >> 8) & 0x00ff) | ((buf[i] << 8) & 0xff00);
|
2006-01-02 17:28:59 +00:00
|
|
|
}
|
2006-01-02 18:31:50 +00:00
|
|
|
}
|