⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 gsm.c

📁 基于sip协议的网络电话源码
💻 C
📖 第 1 页 / 共 2 页
字号:
    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 + -