📄 gsm.c
字号:
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(gsm_codec_factory.pool, sizeof(pjmedia_codec)); PJ_ASSERT_RETURN(codec != NULL, PJ_ENOMEM); codec->op = &gsm_op; codec->factory = factory; gsm_data = pj_pool_zalloc(gsm_codec_factory.pool, sizeof(struct gsm_data)); codec->codec_data = gsm_data; /* 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; } /* 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; pj_int16_t frame[160]; int i; PJ_ASSERT_RETURN(factory && codec, PJ_EINVAL); PJ_ASSERT_RETURN(factory == &gsm_codec_factory.base, PJ_EINVAL); gsm_data = codec->codec_data; /* Close codec, if it's not closed. */ gsm_codec_close(codec); /* Clear left samples in the PLC, since codec+plc will be reused * next time. */ for (i=0; i<2; ++i) { pjmedia_zero_samples(frame, PJ_ARRAY_SIZE(frame)); pjmedia_plc_save(gsm_data->plc, frame); } /* 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 = 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 = 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 = 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 = 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, 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 = 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 (gsm_data->plc_enabled) pjmedia_plc_save( gsm_data->plc, output->buf); return PJ_SUCCESS;}/* * 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 /* PJMEDIA_HAS_GSM_CODEC */
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -