📄 gsm.c
字号:
struct gsm_data *gsm_data;
pj_status_t status;
PJ_ASSERT_RETURN(factory && id && p_codec, PJ_EINVAL);
PJ_ASSERT_RETURN(factory == &gsm_codec_factory.base, PJ_EINVAL);
pj_mutex_lock(gsm_codec_factory.mutex);
/* Get free nodes, if any. */
if (!pj_list_empty(&gsm_codec_factory.codec_list)) {
codec = gsm_codec_factory.codec_list.next;
pj_list_erase(codec);
} else {
codec = PJ_POOL_ZALLOC_T(gsm_codec_factory.pool, pjmedia_codec);
PJ_ASSERT_RETURN(codec != NULL, PJ_ENOMEM);
codec->op = &gsm_op;
codec->factory = factory;
gsm_data = PJ_POOL_ZALLOC_T(gsm_codec_factory.pool, struct gsm_data);
codec->codec_data = gsm_data;
#if !PLC_DISABLED
/* Create PLC */
status = pjmedia_plc_create(gsm_codec_factory.pool, 8000,
160, 0, &gsm_data->plc);
if (status != PJ_SUCCESS) {
pj_mutex_unlock(gsm_codec_factory.mutex);
return status;
}
#endif
/* Create silence detector */
status = pjmedia_silence_det_create(gsm_codec_factory.pool,
8000, 160,
&gsm_data->vad);
if (status != PJ_SUCCESS) {
pj_mutex_unlock(gsm_codec_factory.mutex);
return status;
}
}
pj_mutex_unlock(gsm_codec_factory.mutex);
*p_codec = codec;
return PJ_SUCCESS;
}
/*
* Free codec.
*/
static pj_status_t gsm_dealloc_codec( pjmedia_codec_factory *factory,
pjmedia_codec *codec )
{
struct gsm_data *gsm_data;
int i;
PJ_ASSERT_RETURN(factory && codec, PJ_EINVAL);
PJ_ASSERT_RETURN(factory == &gsm_codec_factory.base, PJ_EINVAL);
gsm_data = (struct gsm_data*) codec->codec_data;
/* Close codec, if it's not closed. */
gsm_codec_close(codec);
#if !PLC_DISABLED
/* Clear left samples in the PLC, since codec+plc will be reused
* next time.
*/
for (i=0; i<2; ++i) {
pj_int16_t frame[160];
pjmedia_zero_samples(frame, PJ_ARRAY_SIZE(frame));
pjmedia_plc_save(gsm_data->plc, frame);
}
#else
PJ_UNUSED_ARG(i);
#endif
/* Re-init silence_period */
pj_set_timestamp32(&gsm_data->last_tx, 0, 0);
/* Put in the free list. */
pj_mutex_lock(gsm_codec_factory.mutex);
pj_list_push_front(&gsm_codec_factory.codec_list, codec);
pj_mutex_unlock(gsm_codec_factory.mutex);
return PJ_SUCCESS;
}
/*
* Init codec.
*/
static pj_status_t gsm_codec_init( pjmedia_codec *codec,
pj_pool_t *pool )
{
PJ_UNUSED_ARG(codec);
PJ_UNUSED_ARG(pool);
return PJ_SUCCESS;
}
/*
* Open codec.
*/
static pj_status_t gsm_codec_open( pjmedia_codec *codec,
pjmedia_codec_param *attr )
{
struct gsm_data *gsm_data = (struct gsm_data*) codec->codec_data;
pj_assert(gsm_data != NULL);
pj_assert(gsm_data->encoder == NULL && gsm_data->decoder == NULL);
gsm_data->encoder = gsm_create();
if (!gsm_data->encoder)
return PJMEDIA_CODEC_EFAILED;
gsm_data->decoder = gsm_create();
if (!gsm_data->decoder)
return PJMEDIA_CODEC_EFAILED;
gsm_data->vad_enabled = (attr->setting.vad != 0);
gsm_data->plc_enabled = (attr->setting.plc != 0);
return PJ_SUCCESS;
}
/*
* Close codec.
*/
static pj_status_t gsm_codec_close( pjmedia_codec *codec )
{
struct gsm_data *gsm_data = (struct gsm_data*) codec->codec_data;
pj_assert(gsm_data != NULL);
if (gsm_data->encoder) {
gsm_destroy(gsm_data->encoder);
gsm_data->encoder = NULL;
}
if (gsm_data->decoder) {
gsm_destroy(gsm_data->decoder);
gsm_data->decoder = NULL;
}
return PJ_SUCCESS;
}
/*
* Modify codec settings.
*/
static pj_status_t gsm_codec_modify(pjmedia_codec *codec,
const pjmedia_codec_param *attr )
{
struct gsm_data *gsm_data = (struct gsm_data*) codec->codec_data;
pj_assert(gsm_data != NULL);
pj_assert(gsm_data->encoder != NULL && gsm_data->decoder != NULL);
gsm_data->vad_enabled = (attr->setting.vad != 0);
gsm_data->plc_enabled = (attr->setting.plc != 0);
return PJ_SUCCESS;
}
/*
* Get frames in the packet.
*/
static pj_status_t gsm_codec_parse( pjmedia_codec *codec,
void *pkt,
pj_size_t pkt_size,
const pj_timestamp *ts,
unsigned *frame_cnt,
pjmedia_frame frames[])
{
unsigned count = 0;
PJ_UNUSED_ARG(codec);
PJ_ASSERT_RETURN(frame_cnt, PJ_EINVAL);
while (pkt_size >= 33 && count < *frame_cnt) {
frames[count].type = PJMEDIA_FRAME_TYPE_AUDIO;
frames[count].buf = pkt;
frames[count].size = 33;
frames[count].timestamp.u64 = ts->u64 + count * 160;
pkt = ((char*)pkt) + 33;
pkt_size -= 33;
++count;
}
*frame_cnt = count;
return PJ_SUCCESS;
}
/*
* Encode frame.
*/
static pj_status_t gsm_codec_encode( pjmedia_codec *codec,
const struct pjmedia_frame *input,
unsigned output_buf_len,
struct pjmedia_frame *output)
{
struct gsm_data *gsm_data = (struct gsm_data*) codec->codec_data;
pj_assert(gsm_data != NULL);
PJ_ASSERT_RETURN(input && output, PJ_EINVAL);
if (output_buf_len < 33)
return PJMEDIA_CODEC_EFRMTOOSHORT;
PJ_ASSERT_RETURN(input->size==320, PJMEDIA_CODEC_EPCMFRMINLEN);
/* Detect silence */
if (gsm_data->vad_enabled) {
pj_bool_t is_silence;
pj_int32_t silence_duration;
silence_duration = pj_timestamp_diff32(&gsm_data->last_tx,
&input->timestamp);
is_silence = pjmedia_silence_det_detect(gsm_data->vad,
(const pj_int16_t*) input->buf,
(input->size >> 1),
NULL);
if (is_silence &&
PJMEDIA_CODEC_MAX_SILENCE_PERIOD != -1 &&
silence_duration < PJMEDIA_CODEC_MAX_SILENCE_PERIOD)
{
output->type = PJMEDIA_FRAME_TYPE_NONE;
output->buf = NULL;
output->size = 0;
output->timestamp = input->timestamp;
return PJ_SUCCESS;
} else {
gsm_data->last_tx = input->timestamp;
}
}
/* Encode */
gsm_encode(gsm_data->encoder, (short*)input->buf,
(unsigned char*)output->buf);
output->size = 33;
output->type = PJMEDIA_FRAME_TYPE_AUDIO;
return PJ_SUCCESS;
}
/*
* Decode frame.
*/
static pj_status_t gsm_codec_decode( pjmedia_codec *codec,
const struct pjmedia_frame *input,
unsigned output_buf_len,
struct pjmedia_frame *output)
{
struct gsm_data *gsm_data = (struct gsm_data*) codec->codec_data;
pj_assert(gsm_data != NULL);
PJ_ASSERT_RETURN(input && output, PJ_EINVAL);
if (output_buf_len < 320)
return PJMEDIA_CODEC_EPCMTOOSHORT;
if (input->size < 33)
return PJMEDIA_CODEC_EFRMTOOSHORT;
gsm_decode(gsm_data->decoder,
(unsigned char*)input->buf,
(short*)output->buf);
output->size = 320;
output->type = PJMEDIA_FRAME_TYPE_AUDIO;
#if !PLC_DISABLED
if (gsm_data->plc_enabled)
pjmedia_plc_save( gsm_data->plc, output->buf);
#endif
return PJ_SUCCESS;
}
#if !PLC_DISABLED
/*
* Recover lost frame.
*/
static pj_status_t gsm_codec_recover(pjmedia_codec *codec,
unsigned output_buf_len,
struct pjmedia_frame *output)
{
struct gsm_data *gsm_data = codec->codec_data;
PJ_ASSERT_RETURN(gsm_data->plc_enabled, PJ_EINVALIDOP);
PJ_ASSERT_RETURN(output_buf_len >= 320, PJMEDIA_CODEC_EPCMTOOSHORT);
pjmedia_plc_generate(gsm_data->plc, output->buf);
output->size = 320;
return PJ_SUCCESS;
}
#endif
#endif /* PJMEDIA_HAS_GSM_CODEC */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -