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

📄 vorbis.c

📁 video linux conference
💻 C
📖 第 1 页 / 共 2 页
字号:
    if( !aout_DateGet( &p_sys->end_date ) )    {        /* We've just started the stream, wait for the first PTS. */        if( p_block ) block_Release( p_block );        return NULL;    }    *pp_block = NULL; /* To avoid being fed the same packet again */    if( p_sys->b_packetizer )    {        return SendPacket( p_dec, p_oggpacket, p_block );    }    else    {        aout_buffer_t *p_aout_buffer;        if( p_sys->i_headers >= 3 )            p_aout_buffer = DecodePacket( p_dec, p_oggpacket );        else            p_aout_buffer = NULL;        if( p_block ) block_Release( p_block );        return p_aout_buffer;    }}/***************************************************************************** * DecodePacket: decodes a Vorbis packet. *****************************************************************************/static aout_buffer_t *DecodePacket( decoder_t *p_dec, ogg_packet *p_oggpacket ){    decoder_sys_t *p_sys = p_dec->p_sys;    int           i_samples;#ifdef MODULE_NAME_IS_tremor    int32_t       **pp_pcm;#else    float         **pp_pcm;#endif    if( p_oggpacket->bytes &&#ifdef MODULE_NAME_IS_tremor        vorbis_synthesis( &p_sys->vb, p_oggpacket, 1 ) == 0 )#else        vorbis_synthesis( &p_sys->vb, p_oggpacket ) == 0 )#endif        vorbis_synthesis_blockin( &p_sys->vd, &p_sys->vb );    /* **pp_pcm is a multichannel float vector. In stereo, for     * example, pp_pcm[0] is left, and pp_pcm[1] is right. i_samples is     * the size of each channel. Convert the float values     * (-1.<=range<=1.) to whatever PCM format and write it out */    if( ( i_samples = vorbis_synthesis_pcmout( &p_sys->vd, &pp_pcm ) ) > 0 )    {        aout_buffer_t *p_aout_buffer;        p_aout_buffer =            p_dec->pf_aout_buffer_new( p_dec, i_samples );        if( p_aout_buffer == NULL ) return NULL;        /* Interleave the samples */#ifdef MODULE_NAME_IS_tremor        Interleave( (int32_t *)p_aout_buffer->p_buffer,                    (const int32_t **)pp_pcm, p_sys->vi.channels, i_samples );#else        Interleave( (float *)p_aout_buffer->p_buffer,                    (const float **)pp_pcm, p_sys->vi.channels, i_samples );#endif        /* Tell libvorbis how many samples we actually consumed */        vorbis_synthesis_read( &p_sys->vd, i_samples );        /* Date management */        p_aout_buffer->start_date = aout_DateGet( &p_sys->end_date );        p_aout_buffer->end_date = aout_DateIncrement( &p_sys->end_date,                                                      i_samples );        return p_aout_buffer;    }    else    {        return NULL;    }}/***************************************************************************** * SendPacket: send an ogg dated packet to the stream output. *****************************************************************************/static block_t *SendPacket( decoder_t *p_dec, ogg_packet *p_oggpacket,                            block_t *p_block ){    decoder_sys_t *p_sys = p_dec->p_sys;    int i_block_size, i_samples;    i_block_size = vorbis_packet_blocksize( &p_sys->vi, p_oggpacket );    if( i_block_size < 0 ) i_block_size = 0; /* non audio packet */    i_samples = ( p_sys->i_last_block_size + i_block_size ) >> 2;    p_sys->i_last_block_size = i_block_size;    /* Date management */    p_block->i_dts = p_block->i_pts = aout_DateGet( &p_sys->end_date );    if( p_sys->i_headers >= 3 )        p_block->i_length = aout_DateIncrement( &p_sys->end_date, i_samples ) -            p_block->i_pts;    else        p_block->i_length = 0;    return p_block;}/***************************************************************************** * ParseVorbisComments: FIXME should be done in demuxer *****************************************************************************/static void ParseVorbisComments( decoder_t *p_dec ){    input_thread_t *p_input = (input_thread_t *)p_dec->p_parent;    char *psz_name, *psz_value, *psz_comment;    int i = 0;    if( p_input->i_object_type != VLC_OBJECT_INPUT ) return;    while( i < p_dec->p_sys->vc.comments )    {        psz_comment = strdup( p_dec->p_sys->vc.user_comments[i] );        if( !psz_comment )        {            msg_Warn( p_dec, "out of memory" );            break;        }        psz_name = psz_comment;        psz_value = strchr( psz_comment, '=' );        if( psz_value )        {            *psz_value = '\0';            psz_value++;            input_Control( p_input, INPUT_ADD_INFO, _("Vorbis comment"),                           psz_name, psz_value );            /* HACK, we should use meta */            if( strstr( psz_name, "artist" ) )            {                input_Control( p_input, INPUT_ADD_INFO, _("Meta-information"),                               _("Artist"), psz_value );            }            else if( strstr( psz_name, "title" ) )            {                p_input->input.p_item->psz_name = strdup( psz_value );            }        }        /* FIXME */        var_SetInteger( p_input, "item-change", p_input->input.p_item->i_id );        free( psz_comment );        i++;    }}/***************************************************************************** * Interleave: helper function to interleave channels *****************************************************************************/#ifdef MODULE_NAME_IS_tremorstatic void Interleave( int32_t *p_out, const int32_t **pp_in,                        int i_nb_channels, int i_samples ){    int i, j;    for ( j = 0; j < i_samples; j++ )        for ( i = 0; i < i_nb_channels; i++ )            p_out[j * i_nb_channels + i] = pp_in[i][j] * (FIXED32_ONE >> 24);}#elsestatic void Interleave( float *p_out, const float **pp_in,                        int i_nb_channels, int i_samples ){    int i, j;    for ( j = 0; j < i_samples; j++ )        for ( i = 0; i < i_nb_channels; i++ )            p_out[j * i_nb_channels + i] = pp_in[i][j];}#endif/***************************************************************************** * CloseDecoder: vorbis decoder destruction *****************************************************************************/static void CloseDecoder( vlc_object_t *p_this ){    decoder_t *p_dec = (decoder_t *)p_this;    decoder_sys_t *p_sys = p_dec->p_sys;    if( !p_sys->b_packetizer && p_sys->i_headers >= 3 )    {        vorbis_block_clear( &p_sys->vb );        vorbis_dsp_clear( &p_sys->vd );    }    vorbis_comment_clear( &p_sys->vc );    vorbis_info_clear( &p_sys->vi );  /* must be called last */    free( p_sys );}#if defined(HAVE_VORBIS_VORBISENC_H) && !defined(MODULE_NAME_IS_tremor)/***************************************************************************** * encoder_sys_t : vorbis encoder descriptor *****************************************************************************/struct encoder_sys_t{    /*     * Vorbis properties     */    vorbis_info      vi; /* struct that stores all the static vorbis bitstream                            settings */    vorbis_comment   vc; /* struct that stores all the bitstream user                          * comments */    vorbis_dsp_state vd; /* central working state for the packet->PCM                          * decoder */    vorbis_block     vb; /* local working space for packet->PCM decode */    int i_last_block_size;    int i_samples_delay;    int i_channels;    /*     * Common properties     */    mtime_t i_pts;};/***************************************************************************** * OpenEncoder: probe the encoder and return score *****************************************************************************/static int OpenEncoder( vlc_object_t *p_this ){    encoder_t *p_enc = (encoder_t *)p_this;    encoder_sys_t *p_sys;    int i_quality, i_min_bitrate, i_max_bitrate, i;    ogg_packet header[3];    vlc_value_t val;    uint8_t *p_extra;    if( p_enc->fmt_out.i_codec != VLC_FOURCC('v','o','r','b') &&        !p_enc->b_force )    {        return VLC_EGENERIC;    }    /* Allocate the memory needed to store the decoder's structure */    if( ( p_sys = (encoder_sys_t *)malloc(sizeof(encoder_sys_t)) ) == NULL )    {        msg_Err( p_enc, "out of memory" );        return VLC_EGENERIC;    }    p_enc->p_sys = p_sys;    p_enc->pf_encode_audio = Encode;    p_enc->fmt_in.i_codec = VLC_FOURCC('f','l','3','2');    p_enc->fmt_out.i_codec = VLC_FOURCC('v','o','r','b');    sout_CfgParse( p_enc, ENC_CFG_PREFIX, ppsz_enc_options, p_enc->p_cfg );    var_Get( p_enc, ENC_CFG_PREFIX "quality", &val );    i_quality = val.i_int;    if( i_quality > 10 ) i_quality = 10;    if( i_quality < 0 ) i_quality = 0;    var_Get( p_enc, ENC_CFG_PREFIX "cbr", &val );    if( val.b_bool ) i_quality = 0;    var_Get( p_enc, ENC_CFG_PREFIX "max-bitrate", &val );    i_max_bitrate = val.i_int;    var_Get( p_enc, ENC_CFG_PREFIX "min-bitrate", &val );    i_min_bitrate = val.i_int;    /* Initialize vorbis encoder */    vorbis_info_init( &p_sys->vi );    if( i_quality > 0 )    {        /* VBR mode */        if( vorbis_encode_setup_vbr( &p_sys->vi,              p_enc->fmt_in.audio.i_channels, p_enc->fmt_in.audio.i_rate,              i_quality * 0.1 ) )        {            vorbis_info_clear( &p_sys->vi );            free( p_enc->p_sys );            msg_Err( p_enc, "VBR mode initialisation failed" );            return VLC_EGENERIC;        }        /* Do we have optional hard quality restrictions? */        if( i_max_bitrate > 0 || i_min_bitrate > 0 )        {            struct ovectl_ratemanage_arg ai;            vorbis_encode_ctl( &p_sys->vi, OV_ECTL_RATEMANAGE_GET, &ai );            ai.bitrate_hard_min = i_min_bitrate;            ai.bitrate_hard_max = i_max_bitrate;            ai.management_active = 1;            vorbis_encode_ctl( &p_sys->vi, OV_ECTL_RATEMANAGE_SET, &ai );        }        else        {            /* Turn off management entirely */            vorbis_encode_ctl( &p_sys->vi, OV_ECTL_RATEMANAGE_SET, NULL );        }    }    else    {        if( vorbis_encode_setup_managed( &p_sys->vi,              p_enc->fmt_in.audio.i_channels, p_enc->fmt_in.audio.i_rate,              i_min_bitrate > 0 ? i_min_bitrate * 1000: -1,              p_enc->fmt_out.i_bitrate,              i_max_bitrate > 0 ? i_max_bitrate * 1000: -1 ) )          {              vorbis_info_clear( &p_sys->vi );              msg_Err( p_enc, "CBR mode initialisation failed" );              free( p_enc->p_sys );              return VLC_EGENERIC;          }    }    vorbis_encode_setup_init( &p_sys->vi );    /* Add a comment */    vorbis_comment_init( &p_sys->vc);    vorbis_comment_add_tag( &p_sys->vc, "ENCODER", "VLC media player");    /* Set up the analysis state and auxiliary encoding storage */    vorbis_analysis_init( &p_sys->vd, &p_sys->vi );    vorbis_block_init( &p_sys->vd, &p_sys->vb );    /* Create and store headers */    vorbis_analysis_headerout( &p_sys->vd, &p_sys->vc,                               &header[0], &header[1], &header[2]);    p_enc->fmt_out.i_extra = 3 * 2 + header[0].bytes +       header[1].bytes + header[2].bytes;    p_extra = p_enc->fmt_out.p_extra = malloc( p_enc->fmt_out.i_extra );    for( i = 0; i < 3; i++ )    {        *(p_extra++) = header[i].bytes >> 8;        *(p_extra++) = header[i].bytes & 0xFF;        memcpy( p_extra, header[i].packet, header[i].bytes );        p_extra += header[i].bytes;    }    p_sys->i_channels = p_enc->fmt_in.audio.i_channels;    p_sys->i_last_block_size = 0;    p_sys->i_samples_delay = 0;    p_sys->i_pts = 0;    return VLC_SUCCESS;}/**************************************************************************** * Encode: the whole thing **************************************************************************** * This function spits out ogg packets. ****************************************************************************/static block_t *Encode( encoder_t *p_enc, aout_buffer_t *p_aout_buf ){    encoder_sys_t *p_sys = p_enc->p_sys;    ogg_packet oggpacket;    block_t *p_block, *p_chain = NULL;    float **buffer;    int i;    unsigned int j;    p_sys->i_pts = p_aout_buf->start_date -                (mtime_t)1000000 * (mtime_t)p_sys->i_samples_delay /                (mtime_t)p_enc->fmt_in.audio.i_rate;    p_sys->i_samples_delay += p_aout_buf->i_nb_samples;    buffer = vorbis_analysis_buffer( &p_sys->vd, p_aout_buf->i_nb_samples );    /* convert samples to float and uninterleave */    for( i = 0; i < p_sys->i_channels; i++ )    {        for( j = 0 ; j < p_aout_buf->i_nb_samples ; j++ )        {            buffer[i][j]= ((float *)p_aout_buf->p_buffer)                                    [j * p_sys->i_channels + i ];        }    }    vorbis_analysis_wrote( &p_sys->vd, p_aout_buf->i_nb_samples );    while( vorbis_analysis_blockout( &p_sys->vd, &p_sys->vb ) == 1 )    {        int i_samples;        vorbis_analysis( &p_sys->vb, NULL );        vorbis_bitrate_addblock( &p_sys->vb );        while( vorbis_bitrate_flushpacket( &p_sys->vd, &oggpacket ) )        {            int i_block_size;            p_block = block_New( p_enc, oggpacket.bytes );            memcpy( p_block->p_buffer, oggpacket.packet, oggpacket.bytes );            i_block_size = vorbis_packet_blocksize( &p_sys->vi, &oggpacket );            if( i_block_size < 0 ) i_block_size = 0;            i_samples = ( p_sys->i_last_block_size + i_block_size ) >> 2;            p_sys->i_last_block_size = i_block_size;            p_block->i_length = (mtime_t)1000000 *                (mtime_t)i_samples / (mtime_t)p_enc->fmt_in.audio.i_rate;            p_block->i_dts = p_block->i_pts = p_sys->i_pts;            p_sys->i_samples_delay -= i_samples;            /* Update pts */            p_sys->i_pts += p_block->i_length;            block_ChainAppend( &p_chain, p_block );        }    }    return p_chain;}/***************************************************************************** * CloseEncoder: vorbis encoder destruction *****************************************************************************/static void CloseEncoder( vlc_object_t *p_this ){    encoder_t *p_enc = (encoder_t *)p_this;    encoder_sys_t *p_sys = p_enc->p_sys;    vorbis_block_clear( &p_sys->vb );    vorbis_dsp_clear( &p_sys->vd );    vorbis_comment_clear( &p_sys->vc );    vorbis_info_clear( &p_sys->vi );  /* must be called last */    free( p_sys );}#endif /* HAVE_VORBIS_VORBISENC_H && !MODULE_NAME_IS_tremor */

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -