14年用FS的这套G729编码,和腾讯QQ内测Silk编码进行转码测试,IBM 16核心32G,做到500并发。
由于G729算法的专利保护,以下代码仅供学习参考,欢迎加本人微信交流。
mod_g729.c
/*
*
* The g729 codec itself is not distributed with this module.
*
* mod_g729.c -- G.729 Codec Module
*
*/
#include "switch.h"
SWITCH_MODULE_LOAD_FUNCTION(mod_g729_load);
SWITCH_MODULE_DEFINITION(mod_g729, mod_g729_load, NULL, NULL);
#define SLIN_FRAME_LEN 160
#define G729_FRAME_LEN 10
#define G729_SAMPLES 80 /* 10ms at 8000 hz, 160 bytes signed linear */
#define BUFFER_SAMPLES 8000
#include "g729a_v11/typedef.h"
#include "g729a_v11/basic_op.h"
#include "g729a_v11/ld8a.h"
#include "g729a_v11/tab_ld8a.h"
#include "g729a_v11/util.h"
#include "g729a_v11/pre_proc.h"
/* Sample frame data Hz zachem ono nado */
#include "slin_g729_ex.h"
#include "g729_slin_ex.h"
struct g72x_coder_pvt {
CodState *coder;
DecState *decoder;
void *scratch_mem;
int16_t buf[BUFFER_SAMPLES]; /* 1 second */
};
#define dec_state g72x_coder_pvt
#define cod_state g72x_coder_pvt
#define PVT struct g72x_coder_pvt
#define g729_coder_pvt translator_pvt
void g729_init_lib(){}
void g729_init_coder(PVT *hEncoder, int dummy){
struct g72x_coder_pvt *state = hEncoder;
// state = malloc(sizeof(struct g72x_coder_pvt));
state->coder = Init_Coder_ld8a();
Init_Pre_Process(state->coder);
return;
}
void g729_release_coder(PVT *hEncoder){
struct g72x_coder_pvt *state = hEncoder;
free (state->coder);
free (state->scratch_mem);
}
void g729_init_decoder(PVT *hDecoder){
struct g72x_coder_pvt *state = hDecoder;
// state = malloc(sizeof(struct g72x_coder_pvt));
state->decoder = Init_Decod_ld8a();
Init_Post_Filter(state->decoder);
Init_Post_Process(state->decoder);
return;
}
void g729_release_decoder(PVT *hDecoder){
struct g72x_coder_pvt *state = hDecoder;
free (state->decoder);
free (state->scratch_mem);
}
void g729_coder(PVT *hEncoder, short *ddp, char *edp, int *cbret){
Word16 parm[PRM_SIZE];
Copy ((Word16 *) ddp, hEncoder->coder->new_speech, 80);
Pre_Process(hEncoder->coder, hEncoder->coder->new_speech, 80);
Coder_ld8a(hEncoder->coder, parm);
Store_Params(parm, edp);
}
/* By Xinke*/
//static int g729_frame_type(int datalen)
//{
// switch (datalen) {
// case 0: return -1; /* erased */
// /* case 0: return 0; maybe it should be 0 - untransmitted silence? */
// case 2: return 1; /* SID */
// case 8: return 2; /* 729d */
// case 10: return 3; /* 729, 729a */
// case 15: return 4; /* 729e */
// }
// return 0;
//}
void g729_decoder(PVT *hDecoder, short *ddp, char *edp, int plen){
// EasyG729A_decoder(*hDecoder, (unsigned char *)edp, (short *)ddp);
//apiG729Decode(hDecoder->coder, edp, g729_frame_type(plen), ddp);
Word16 i;
Word16 *synth;
Word16 parm[PRM_SIZE + 1];
Restore_Params(edp, &parm[1]);
synth = hDecoder->decoder->synth_buf + M;
parm[0] = 1;
for (i = 0; i < PRM_SIZE; i++) {
if (parm[i + 1] != 0) {
parm[0] = 0;
break;
}
}
parm[4] = Check_Parity_Pitch(parm[3], parm[4]);
Decod_ld8a(hDecoder->decoder, parm, synth, hDecoder->decoder->Az_dec,
hDecoder->decoder->T2, &hDecoder->decoder->bad_lsf);
Post_Filter(hDecoder->decoder, synth, hDecoder->decoder->Az_dec, hDecoder->decoder->T2);
Post_Process(hDecoder->decoder, synth, L_FRAME);
memmove( ddp, synth, 2 * L_FRAME);
}
struct g729_context {
struct dec_state decoder_object;
struct cod_state encoder_object;
};
static switch_status_t switch_g729_init(switch_codec_t *codec, switch_codec_flag_t flags, const switch_codec_settings_t *codec_settings)
{
struct g729_context *context = NULL;
int encoding, decoding;
encoding = (flags & SWITCH_CODEC_FLAG_ENCODE);
decoding = (flags & SWITCH_CODEC_FLAG_DECODE);
if (!(encoding || decoding) || (!(context = switch_core_alloc(codec->memory_pool, sizeof(struct g729_context))))) {
return SWITCH_STATUS_FALSE;
} else {
if (codec->fmtp_in) {
codec->fmtp_out = switch_core_strdup(codec->memory_pool, codec->fmtp_in);
}
if (encoding) {
g729_init_coder(&context->encoder_object, 0);
}
if (decoding) {
g729_init_decoder(&context->decoder_object);
}
codec->private_info = context;
return SWITCH_STATUS_SUCCESS;
}
}
static switch_status_t switch_g729_destroy(switch_codec_t *codec)
{
struct g729_context *context;
context = codec->private_info;
g729_release_coder( &(context->encoder_object));
g729_release_decoder( &(context->decoder_object));
codec->private_info = NULL;
return SWITCH_STATUS_SUCCESS;
}
static switch_status_t switch_g729_encode(switch_codec_t *codec,
switch_codec_t *other_codec,
void *decoded_data,
uint32_t decoded_data_len, uint32_t decoded_rate,
void *encoded_data, uint32_t *encoded_data_len, uint32_t *encoded_rate,
unsigned int *flag)
{
struct g729_context *context = codec->private_info;
int cbret = 0;
// switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "g729 encode!!!\n");
if (!context) {
return SWITCH_STATUS_FALSE;
}
if (decoded_data_len % 160 == 0) {
uint32_t new_len = 0;
int16_t *ddp = decoded_data;
char *edp = encoded_data;
int x;
int loops = (int) decoded_data_len / 160;
for (x = 0; x < loops && new_len < *encoded_data_len; x++) {
g729_coder(&context->encoder_object, ddp, edp, &cbret);
edp += 10;
ddp += 80;
new_len += 10;
}
if (new_len <= *encoded_data_len) {
*encoded_data_len = new_len;
} else {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "buffer overflow!!! %u >= %u\n", new_len, *encoded_data_len);
return SWITCH_STATUS_FALSE;
}
}
return SWITCH_STATUS_SUCCESS;
}
// For zero data
static int16_t lost_frame[80] = { 0 };
static switch_status_t switch_g729_decode(switch_codec_t *codec,
switch_codec_t *other_codec,
void *encoded_data,
uint32_t encoded_data_len, uint32_t encoded_rate,
void *decoded_data, uint32_t *decoded_data_len, uint32_t *decoded_rate,
unsigned int *flag)
{
//////////////////// Start code from * g729
int framesize;
int x;
uint32_t new_len = 0;
char *edp = encoded_data;
short *ddp = decoded_data;
struct g729_context *context = codec->private_info;
if (!context) {
return SWITCH_STATUS_FALSE;
}
if (encoded_data_len == 0) { /* Native PLC interpolation */
g729_decoder(&context->decoder_object, ddp,(char *)lost_frame, 0);
ddp+=80;
decoded_data_len=(uint32_t *)160;
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "g729 zero length frame\n");
return SWITCH_STATUS_SUCCESS;
}
for(x = 0; x < encoded_data_len && new_len < *decoded_data_len; x += framesize) {
if(encoded_data_len - x < 8)
framesize = 2; /* SID */
else
framesize = 10; /* regular 729a frame */
g729_decoder(&context->decoder_object, ddp, edp, framesize);
ddp += 80;
edp += framesize;
new_len += 160;
}
if (new_len <= *decoded_data_len) {
*decoded_data_len = new_len;
} else {
switch_log_printf(SWITCH_CHANNEL_LOG, SWITCH_LOG_ERROR, "buffer overflow!!!\n");
return SWITCH_STATUS_FALSE;
}
return SWITCH_STATUS_SUCCESS;
//////////////////// End code from * g729
}
SWITCH_MODULE_LOAD_FUNCTION(mod_g729_load)
{
switch_codec_interface_t *codec_interface;
int mpf = 10000, spf = 80, bpf = 160, ebpf = 10, count;
/* connect my internal structure to the blank pointer passed to me */
*module_interface = switch_loadable_module_create_module_interface(pool, modname);
SWITCH_ADD_CODEC(codec_interface, "G.729");
for (count = 12; count > 0; count--) {
switch_core_codec_add_implementation(pool, codec_interface,
SWITCH_CODEC_TYPE_AUDIO, 18, "G729", NULL, 8000, 8000, 8000,
mpf * count, spf * count, bpf * count, ebpf * count, 1, count * 10,
switch_g729_init, switch_g729_encode, switch_g729_decode, switch_g729_destroy);
}
/* indicate that the module should continue to be loaded */
return SWITCH_STATUS_SUCCESS;
}
/* For Emacs:
* Local Variables:
* mode:c
* indent-tabs-mode:t
* tab-width:4
* c-basic-offset:4
* End:
* For VIM:
* vim:set softtabstop=4 shiftwidth=4 tabstop=4:
*/codec_g729.c
/*
* G729 codec for Asterisk
*
* For G.729(,A,B) royalty payments, see http://www.sipro.com
*
* WARNING: please make sure you are sitting down before looking
* at their price list.
*
* This source file is Copyright (C) 2004 Ready Technology Limited
* This code is provided for educational purposes and is not warranted
* to be fit for commercial use. There is no warranty of any kind.
*
* Author: daniel@readytechnology.co.uk
*/
#include "asterisk/lock.h"
#include "asterisk/translate.h"
#include "asterisk/module.h"
#include "asterisk/logger.h"
#include "asterisk/channel.h"
#include <pthread.h>
#include <fcntl.h>
#include <stdlib.h>
#include <unistd.h>
#include <netinet/in.h>
#include <string.h>
#include <stdio.h>
#include "g729a_v11/typedef.h"
#include "g729a_v11/basic_op.h"
#include "g729a_v11/ld8a.h"
#include "g729a_v11/tab_ld8a.h"
#include "g729a_v11/util.h"
#include "g729a_v11/pre_proc.h"
/* Sample frame data */
#include "slin_g729_ex.h"
#include "g729_slin_ex.h"
AST_MUTEX_DEFINE_STATIC(localuser_lock);
static int localusecnt=0;
static char *tdesc = "G729/PCM16 (signed linear) Codec Translator, based on ITU Reference code";
struct ast_translator_pvt {
struct ast_frame f;
CodState *coder;
DecState *decoder;
short pcm_buf[8000];
unsigned char bitstream_buf[1000];
int tail;
};
#define g729_coder_pvt ast_translator_pvt
static struct ast_translator_pvt *lintog729_new(void) {
struct g729_coder_pvt *tmp;
tmp = malloc(sizeof(struct g729_coder_pvt));
if(tmp) {
tmp->coder = Init_Coder_ld8a();
Init_Pre_Process(tmp->coder);
tmp->decoder = NULL;
tmp->tail = 0;
localusecnt++;
}
return tmp;
}
static struct ast_translator_pvt *g729tolin_new(void) {
struct g729_coder_pvt *tmp;
tmp = malloc(sizeof(struct g729_coder_pvt));
if(tmp) {
tmp->decoder = Init_Decod_ld8a();
Init_Post_Filter(tmp->decoder);
Init_Post_Process(tmp->decoder);
tmp->coder = NULL;
tmp->tail = 0;
localusecnt++;
}
return tmp;
}
static struct ast_frame *lintog729_sample(void) {
static struct ast_frame f;
f.frametype = AST_FRAME_VOICE;
f.subclass = AST_FORMAT_SLINEAR;
f.datalen = sizeof(slin_g729_ex);
f.samples = sizeof(slin_g729_ex) / 2;
f.mallocd = 0;
f.offset = 0;
f.src = __PRETTY_FUNCTION__;
f.data = slin_g729_ex;
return &f;
}
static struct ast_frame *g729tolin_sample(void) {
static struct ast_frame f;
f.frametype = AST_FRAME_VOICE;
f.subclass = AST_FORMAT_G729A;
f.datalen = sizeof(g729_slin_ex);
f.samples = 240;
f.mallocd = 0;
f.offset = 0;
f.src = __PRETTY_FUNCTION__;
f.data = g729_slin_ex;
return &f;
}
/**
* Retrieve a frame that has already been decompressed
*/
static struct ast_frame *g729tolin_frameout(struct ast_translator_pvt *tmp) {
if(!tmp->tail)
return NULL;
tmp->f.frametype = AST_FRAME_VOICE;
tmp->f.subclass = AST_FORMAT_SLINEAR;
tmp->f.datalen = tmp->tail * 2;
tmp->f.samples = tmp->tail;
tmp->f.mallocd = 0;
tmp->f.offset = AST_FRIENDLY_OFFSET;
tmp->f.src = __PRETTY_FUNCTION__;
tmp->f.data = tmp->pcm_buf;
tmp->tail = 0;
return &tmp->f;
}
/**
* Accept a frame and decode it at the end of the current buffer
*/
static int g729tolin_framein(struct ast_translator_pvt *tmp, struct ast_frame *f) {
int x;
int frameSize = 0;
for(x = 0; x < f->datalen; x += frameSize) {
if((f->datalen - x) == 2)
frameSize = 2; /* VAD frame */
else
frameSize = 10; /* Regular frame */
if(tmp->tail + 80 < sizeof(tmp->pcm_buf) / 2) {
{
Word16 i;
Word16 *synth;
Word16 parm[PRM_SIZE + 1];
Restore_Params(f->data + x, &parm[1]);
synth = tmp->decoder->synth_buf + M;
parm[0] = 1;
for (i = 0; i < PRM_SIZE; i++) {
if (parm[i + 1] != 0) {
parm[0] = 0;
break;
}
}
parm[4] = Check_Parity_Pitch(parm[3], parm[4]);
Decod_ld8a(tmp->decoder, parm, synth, tmp->decoder->Az_dec, tmp->decoder->T2, &tmp->decoder->bad_lsf);
Post_Filter(tmp->decoder, synth, tmp->decoder->Az_dec, tmp->decoder->T2);
Post_Process(tmp->decoder, synth, L_FRAME);
memmove(tmp->pcm_buf + tmp->tail, synth, 2 * L_FRAME);
}
tmp->tail += 80;
} else {
ast_log(LOG_WARNING, "Out of G.729 buffer space\n");
return -1;
}
}
return 0;
}
static int lintog729_framein(struct ast_translator_pvt *tmp, struct ast_frame *f) {
if(tmp->tail + f->datalen/2 < sizeof(tmp->pcm_buf) / 2) {
memcpy((tmp->pcm_buf + tmp->tail), f->data, f->datalen);
tmp->tail += f->datalen/2;
} else {
ast_log(LOG_WARNING, "Out of buffer space\n");
return -1;
}
return 0;
}
static struct ast_frame *lintog729_frameout(struct ast_translator_pvt *tmp) {
int x = 0;
if(tmp->tail < 80)
return NULL;
tmp->f.frametype = AST_FRAME_VOICE;
tmp->f.subclass = AST_FORMAT_G729A;
tmp->f.mallocd = 0;
tmp->f.offset = AST_FRIENDLY_OFFSET;
tmp->f.src = __PRETTY_FUNCTION__;
tmp->f.data = tmp->bitstream_buf;
while(tmp->tail >= 80) {
if((x+1) * 10 >= sizeof(tmp->bitstream_buf)) {
ast_log(LOG_WARNING, "Out of buffer space\n");
break;
}
{
Word16 parm[PRM_SIZE];
Copy ((Word16 *) tmp->pcm_buf, tmp->coder->new_speech, 80);
Pre_Process(tmp->coder, tmp->coder->new_speech, 80);
Coder_ld8a(tmp->coder, parm);
Store_Params(parm, tmp->bitstream_buf + (x * 10));
}
tmp->tail -= 80;
if(tmp->tail)
memmove(tmp->pcm_buf, tmp->pcm_buf + 80, tmp->tail * 2);
x++;
}
tmp->f.datalen = x * 10;
tmp->f.samples = x * 80;
return &(tmp->f);
}
static void g729_release(struct ast_translator_pvt *pvt) {
if (pvt->coder)
free (pvt->coder);
if (pvt->decoder)
free (pvt->decoder);
localusecnt--;
}
static struct ast_translator g729tolin = {
"g729tolin",
AST_FORMAT_G729A, AST_FORMAT_SLINEAR,
g729tolin_new,
g729tolin_framein,
g729tolin_frameout,
g729_release,
g729tolin_sample };
static struct ast_translator lintog729 = {
"lintog729",
AST_FORMAT_SLINEAR, AST_FORMAT_G729A,
lintog729_new,
lintog729_framein,
lintog729_frameout,
g729_release,
lintog729_sample };
int load_module(void) {
int res;
res = ast_register_translator(&g729tolin);
if(!res)
res = ast_register_translator(&lintog729);
else
ast_unregister_translator(&g729tolin);
return res;
}
int unload_module(void) {
int res;
ast_mutex_lock(&localuser_lock);
res = ast_unregister_translator(&lintog729);
if(!res)
res = ast_unregister_translator(&g729tolin);
if(localusecnt)
res = -1;
ast_mutex_unlock(&localuser_lock);
return res;
}
char *description(void) {
return tdesc;
}
int usecount(void) {
int res;
STANDARD_USECOUNT(res);
return res;
}
char *key() {
return ASTERISK_GPL_KEY;
}
发表评论