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

📄 transcode.c

📁 video linux conference
💻 C
📖 第 1 页 / 共 5 页
字号:
    if( id->p_encoder->fmt_out.i_codec == VLC_FOURCC( 'm','p','3',' ' ) )        id->p_encoder->fmt_out.i_codec = VLC_FOURCC( 'm','p','g','a' );    return VLC_SUCCESS;}static void transcode_audio_close( sout_stream_t *p_stream,                                   sout_stream_id_t *id ){    int i;    /* Close decoder */    if( id->p_decoder->p_module )        module_Unneed( id->p_decoder, id->p_decoder->p_module );    id->p_decoder->p_module = 0;    /* Close encoder */    if( id->p_encoder->p_module )        module_Unneed( id->p_encoder, id->p_encoder->p_module );    id->p_encoder->p_module = 0;    /* Close filters */    for( i = 0; i < id->i_filter; i++ )    {        vlc_object_detach( id->pp_filter[i] );        if( id->pp_filter[i]->p_module )            module_Unneed( id->pp_filter[i], id->pp_filter[i]->p_module );        vlc_object_destroy( id->pp_filter[i] );    }}static int transcode_audio_process( sout_stream_t *p_stream,                                    sout_stream_id_t *id,                                    block_t *in, block_t **out ){    sout_stream_sys_t *p_sys = p_stream->p_sys;    aout_buffer_t *p_audio_buf;    block_t *p_block, *p_audio_block;    int i;    *out = NULL;    while( (p_audio_buf = id->p_decoder->pf_decode_audio( id->p_decoder,                                                          &in )) )    {        if( p_sys->b_master_sync )        {            mtime_t i_dts = date_Get( &id->interpolated_pts ) + 1;            p_sys->i_master_drift = p_audio_buf->start_date - i_dts;            date_Increment( &id->interpolated_pts, p_audio_buf->i_nb_samples );            p_audio_buf->start_date -= p_sys->i_master_drift;            p_audio_buf->end_date -= p_sys->i_master_drift;        }        p_audio_block = p_audio_buf->p_sys;        p_audio_block->i_buffer = p_audio_buf->i_nb_bytes;        p_audio_block->i_dts = p_audio_block->i_pts =            p_audio_buf->start_date;        p_audio_block->i_length = p_audio_buf->end_date -            p_audio_buf->start_date;        p_audio_block->i_samples = p_audio_buf->i_nb_samples;        /* Run filter chain */        for( i = 0; i < id->i_filter; i++ )        {            p_audio_block =                id->pp_filter[i]->pf_audio_filter( id->pp_filter[i],                                                   p_audio_block );        }        p_audio_buf->p_buffer = p_audio_block->p_buffer;        p_audio_buf->i_nb_bytes = p_audio_block->i_buffer;        p_audio_buf->i_nb_samples = p_audio_block->i_samples;        p_audio_buf->start_date = p_audio_block->i_dts;        p_audio_buf->end_date = p_audio_block->i_dts + p_audio_block->i_length;        p_block = id->p_encoder->pf_encode_audio( id->p_encoder, p_audio_buf );        block_ChainAppend( out, p_block );        block_Release( p_audio_block );        free( p_audio_buf );    }    return VLC_SUCCESS;}static void audio_release_buffer( aout_buffer_t *p_buffer ){    if( p_buffer && p_buffer->p_sys ) block_Release( p_buffer->p_sys );    if( p_buffer ) free( p_buffer );}static aout_buffer_t *audio_new_buffer( decoder_t *p_dec, int i_samples ){    aout_buffer_t *p_buffer;    block_t *p_block;    int i_size;    if( p_dec->fmt_out.audio.i_bitspersample )    {        i_size = i_samples * p_dec->fmt_out.audio.i_bitspersample / 8 *            p_dec->fmt_out.audio.i_channels;    }    else if( p_dec->fmt_out.audio.i_bytes_per_frame &&             p_dec->fmt_out.audio.i_frame_length )    {        i_size = i_samples * p_dec->fmt_out.audio.i_bytes_per_frame /            p_dec->fmt_out.audio.i_frame_length;    }    else    {        i_size = i_samples * 4 * p_dec->fmt_out.audio.i_channels;    }    p_buffer = malloc( sizeof(aout_buffer_t) );    p_buffer->pf_release = audio_release_buffer;    p_buffer->p_sys = p_block = block_New( p_dec, i_size );    p_buffer->p_buffer = p_block->p_buffer;    p_buffer->i_size = p_buffer->i_nb_bytes = p_block->i_buffer;    p_buffer->i_nb_samples = i_samples;    p_block->i_samples = i_samples;    return p_buffer;}static void audio_del_buffer( decoder_t *p_dec, aout_buffer_t *p_buffer ){    if( p_buffer && p_buffer->p_sys ) block_Release( p_buffer->p_sys );    if( p_buffer ) free( p_buffer );}/* * video */static int transcode_video_new( sout_stream_t *p_stream, sout_stream_id_t *id ){    sout_stream_sys_t *p_sys = p_stream->p_sys;    int i;    /*     * Open decoder     */    /* Initialization of decoder structures */    id->p_decoder->fmt_out = id->p_decoder->fmt_in;    id->p_decoder->fmt_out.i_extra = 0;    id->p_decoder->fmt_out.p_extra = 0;    id->p_decoder->pf_decode_video = 0;    id->p_decoder->pf_vout_buffer_new = video_new_buffer_decoder;    id->p_decoder->pf_vout_buffer_del = video_del_buffer_decoder;    id->p_decoder->pf_picture_link    = video_link_picture_decoder;    id->p_decoder->pf_picture_unlink  = video_unlink_picture_decoder;    id->p_decoder->p_owner = malloc( sizeof(decoder_owner_sys_t) );    for( i = 0; i < PICTURE_RING_SIZE; i++ )        id->p_decoder->p_owner->pp_pics[i] = 0;    id->p_decoder->p_owner->p_sys = p_sys;    //id->p_decoder->p_cfg = p_sys->p_video_cfg;    id->p_decoder->p_module =        module_Need( id->p_decoder, "decoder", "$codec", 0 );    if( !id->p_decoder->p_module )    {        msg_Err( p_stream, "cannot find decoder" );        return VLC_EGENERIC;    }    /*     * Open encoder.     * Because some info about the decoded input will only be available     * once the first frame is decoded, we actually only test the availability     * of the encoder here.     */    /* Initialization of encoder format structures */    es_format_Init( &id->p_encoder->fmt_in, id->p_decoder->fmt_in.i_cat,                    id->p_decoder->fmt_out.i_codec );    id->p_encoder->fmt_in.video.i_chroma = id->p_decoder->fmt_out.i_codec;    /* The dimensions will be set properly later on.     * Just put sensible values so we can test an encoder is available. */    id->p_encoder->fmt_in.video.i_width =        id->p_encoder->fmt_out.video.i_width ?        id->p_encoder->fmt_out.video.i_width :        id->p_decoder->fmt_in.video.i_width ?        id->p_decoder->fmt_in.video.i_width : 16;    id->p_encoder->fmt_in.video.i_height =        id->p_encoder->fmt_out.video.i_height ?        id->p_encoder->fmt_out.video.i_height :        id->p_decoder->fmt_in.video.i_height ?        id->p_decoder->fmt_in.video.i_height : 16;    id->p_encoder->fmt_in.video.i_frame_rate = 25;    id->p_encoder->fmt_in.video.i_frame_rate_base = 1;    id->p_encoder->i_threads = p_sys->i_threads;    id->p_encoder->p_cfg = p_sys->p_video_cfg;    id->p_encoder->p_module =        module_Need( id->p_encoder, "encoder", p_sys->psz_venc, VLC_TRUE );    if( !id->p_encoder->p_module )    {        msg_Err( p_stream, "cannot find encoder" );        module_Unneed( id->p_decoder, id->p_decoder->p_module );        id->p_decoder->p_module = 0;        return VLC_EGENERIC;    }    /* Close the encoder.     * We'll open it only when we have the first frame. */    module_Unneed( id->p_encoder, id->p_encoder->p_module );    id->p_encoder->p_module = NULL;    if( p_sys->i_threads >= 1 )    {        p_sys->id_video = id;        vlc_mutex_init( p_stream, &p_sys->lock_out );        vlc_cond_init( p_stream, &p_sys->cond );        memset( p_sys->pp_pics, 0, sizeof(p_sys->pp_pics) );        p_sys->i_first_pic = 0;        p_sys->i_last_pic = 0;        p_sys->p_buffers = NULL;        p_sys->b_die = p_sys->b_error = 0;        if( vlc_thread_create( p_sys, "encoder", EncoderThread,                               VLC_THREAD_PRIORITY_VIDEO, VLC_FALSE ) )        {            msg_Err( p_stream, "cannot spawn encoder thread" );            module_Unneed( id->p_decoder, id->p_decoder->p_module );            id->p_decoder->p_module = 0;            return VLC_EGENERIC;        }    }    date_Set( &id->interpolated_pts, 0 );    return VLC_SUCCESS;}static int transcode_video_encoder_open( sout_stream_t *p_stream,                                         sout_stream_id_t *id ){    sout_stream_sys_t *p_sys = p_stream->p_sys;    /* Hack because of the copy packetizer which can fail to detect the     * proper size (which forces us to wait until the 1st frame     * is decoded) */    int i_width = id->p_decoder->fmt_out.video.i_width -        p_sys->i_crop_left - p_sys->i_crop_right;    int i_height = id->p_decoder->fmt_out.video.i_height -        p_sys->i_crop_top - p_sys->i_crop_bottom;    if( id->p_encoder->fmt_out.video.i_width <= 0 &&        id->p_encoder->fmt_out.video.i_height <= 0 && p_sys->f_scale )    {        /* Apply the scaling */        id->p_encoder->fmt_out.video.i_width = i_width * p_sys->f_scale;        id->p_encoder->fmt_out.video.i_height = i_height * p_sys->f_scale;    }    else if( id->p_encoder->fmt_out.video.i_width > 0 &&             id->p_encoder->fmt_out.video.i_height <= 0 )    {        id->p_encoder->fmt_out.video.i_height =            id->p_encoder->fmt_out.video.i_width / (double)i_width * i_height;    }    else if( id->p_encoder->fmt_out.video.i_width <= 0 &&             id->p_encoder->fmt_out.video.i_height > 0 )    {        id->p_encoder->fmt_out.video.i_width =            id->p_encoder->fmt_out.video.i_height / (double)i_height * i_width;    }    /* Make sure the size is at least a multiple of 2 */    id->p_encoder->fmt_out.video.i_width =        (id->p_encoder->fmt_out.video.i_width + 1) >> 1 << 1;    id->p_encoder->fmt_out.video.i_height =        (id->p_encoder->fmt_out.video.i_height + 1) >> 1 << 1;    id->p_encoder->fmt_in.video.i_width =        id->p_encoder->fmt_out.video.i_width;    id->p_encoder->fmt_in.video.i_height =        id->p_encoder->fmt_out.video.i_height;    if( !id->p_encoder->fmt_out.video.i_frame_rate ||        !id->p_encoder->fmt_out.video.i_frame_rate_base )    {        if( id->p_decoder->fmt_out.video.i_frame_rate &&            id->p_decoder->fmt_out.video.i_frame_rate_base )        {            id->p_encoder->fmt_out.video.i_frame_rate =                id->p_decoder->fmt_out.video.i_frame_rate;            id->p_encoder->fmt_out.video.i_frame_rate_base =                id->p_decoder->fmt_out.video.i_frame_rate_base;        }        else        {            /* Pick a sensible default value */            id->p_encoder->fmt_out.video.i_frame_rate = 25;            id->p_encoder->fmt_out.video.i_frame_rate_base = 1;        }    }    id->p_encoder->fmt_in.video.i_frame_rate =        id->p_encoder->fmt_out.video.i_frame_rate;    id->p_encoder->fmt_in.video.i_frame_rate_base =        id->p_encoder->fmt_out.video.i_frame_rate_base;    date_Init( &id->interpolated_pts,               id->p_encoder->fmt_out.video.i_frame_rate,               id->p_encoder->fmt_out.video.i_frame_rate_base );    /* Check whether a particular aspect ratio was requested */    if( !id->p_encoder->fmt_out.video.i_aspect )    {        id->p_encoder->fmt_out.video.i_aspect =            id->p_decoder->fmt_out.video.i_aspect;    }    id->p_encoder->fmt_in.video.i_aspect =        id->p_encoder->fmt_out.video.i_aspect;    id->p_encoder->p_module =        module_Need( id->p_encoder, "encoder", p_sys->psz_venc, VLC_TRUE );    if( !id->p_encoder->p_module )    {        msg_Err( p_stream, "cannot find encoder" );        return VLC_EGENERIC;    }    id->p_encoder->fmt_in.video.i_chroma = id->p_encoder->fmt_in.i_codec;    /* Hack for mp2v/mp1v transcoding support */    if( id->p_encoder->fmt_out.i_codec == VLC_FOURCC('m','p','1','v') ||        id->p_encoder->fmt_out.i_codec == VLC_FOURCC('m','p','2','v') )    {        id->p_encoder->fmt_out.i_codec = VLC_FOURCC('m','p','g','v');    }    id->id = p_stream->p_sys->p_out->pf_add( p_stream->p_sys->p_out,                                             &id->p_encoder->fmt_out );    if( !id->id )    {        msg_Err( p_stream, "cannot add this stream" );        return VLC_EGENERIC;    }    return VLC_SUCCESS;}static void transcode_video_close( sout_stream_t *p_stream,                                   sout_stream_id_t *id ){    int i, j;    if( p_stream->p_sys->i_threads >= 1 )    {        vlc_mutex_lock( &p_stream->p_sys->lock_out );        p_stream->p_sys->b_die = 1;        vlc_cond_signal( &p_stream->p_sys->cond );        vlc_mutex_unlock( &p_stream->p_sys->lock_out );        vlc_thread_join( p_stream->p_sys );        vlc_mutex_destroy( &p_stream->p_sys->lock_out );        vlc_cond_destroy( &p_stream->p_sys->cond );    }    /* Close decoder */    if( id->p_decoder->p_module )        module_Unneed( id->p_decoder, id->p_decoder->p_module );    if( id->p_decoder->p_owner )    {        /* Clean-up pictures ring buffer */        for( i = 0; i < PICTURE_RING_SIZE; i++ )        {            if( id->p_decoder->p_owner->pp_pics[i] )                video_del_buffer( VLC_OBJECT(id->p_decoder),                                  id->p_decoder->p_owner->pp_pics[i] );        }        free( id->p_decoder->p_owner );    }

⌨️ 快捷键说明

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