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

📄 mosaic_bridge.c

📁 VLC Player Source Code
💻 C
📖 第 1 页 / 共 2 页
字号:
/***************************************************************************** * mosaic_bridge.c: ***************************************************************************** * Copyright (C) 2004-2007 the VideoLAN team * $Id: a28986ca5608d8f3db91213d7c1aa0fd3db9bddd $ * * Authors: Antoine Cellerier <dionoea@videolan.org> *          Christophe Massiot <massiot@via.ecp.fr> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *****************************************************************************//***************************************************************************** * Preamble *****************************************************************************/#ifdef HAVE_CONFIG_H# include "config.h"#endif#include <errno.h>                                                 /* ENOMEM */#include <vlc_common.h>#include <vlc_plugin.h>#include <vlc_sout.h>#include <vlc_block.h>#include <vlc_codec.h>#include <vlc_image.h>#include <vlc_filter.h>#include "../video_filter/mosaic.h"#include <assert.h>/***************************************************************************** * Local structures *****************************************************************************/struct sout_stream_sys_t{    bridged_es_t *p_es;    vlc_mutex_t *p_lock;    decoder_t       *p_decoder;    image_handler_t *p_image; /* filter for resizing */    int i_height, i_width;    unsigned int i_sar_num, i_sar_den;    char *psz_id;    bool b_inited;    int i_chroma; /* force image format chroma */    filter_chain_t *p_vf2;};#define PICTURE_RING_SIZE 4struct decoder_owner_sys_t{    picture_t *pp_pics[PICTURE_RING_SIZE];    /* Current format in use by the output */    video_format_t video;};typedef void (* pf_release_t)( picture_t * );static void ReleasePicture( picture_t *p_pic ){    assert( p_pic );    if( --p_pic->i_refcount > 0 )        return;    if( p_pic->p_sys )    {        pf_release_t pf_release = (pf_release_t)p_pic->p_sys;        p_pic->p_sys = NULL;        pf_release( p_pic );    }    else    {        free( p_pic->p_data_orig );        free( p_pic );    }}/***************************************************************************** * Local prototypes *****************************************************************************/static int  Open    ( vlc_object_t * );static void Close   ( vlc_object_t * );static sout_stream_id_t *Add ( sout_stream_t *, es_format_t * );static int               Del ( sout_stream_t *, sout_stream_id_t * );static int               Send( sout_stream_t *, sout_stream_id_t *, block_t * );inline static void video_del_buffer_decoder( decoder_t *, picture_t * );inline static void video_del_buffer_filter( filter_t *, picture_t * );static void video_del_buffer( picture_t * );inline static picture_t *video_new_buffer_decoder( decoder_t * );inline static picture_t *video_new_buffer_filter( filter_t * );static picture_t *video_new_buffer( vlc_object_t *, decoder_owner_sys_t *,                                    es_format_t *, void (*)( picture_t * ) );static void video_link_picture_decoder( decoder_t *, picture_t * );static void video_unlink_picture_decoder( decoder_t *, picture_t * );static int HeightCallback( vlc_object_t *, char const *,                           vlc_value_t, vlc_value_t, void * );static int WidthCallback( vlc_object_t *, char const *,                          vlc_value_t, vlc_value_t, void * );static int alphaCallback( vlc_object_t *, char const *,                          vlc_value_t, vlc_value_t, void * );static int xCallback( vlc_object_t *, char const *,                      vlc_value_t, vlc_value_t, void * );static int yCallback( vlc_object_t *, char const *,                      vlc_value_t, vlc_value_t, void * );/***************************************************************************** * Module descriptor *****************************************************************************/#define ID_TEXT N_("ID")#define ID_LONGTEXT N_( \    "Specify an identifier string for this subpicture" )#define WIDTH_TEXT N_("Video width")#define WIDTH_LONGTEXT N_( \    "Output video width." )#define HEIGHT_TEXT N_("Video height")#define HEIGHT_LONGTEXT N_( \    "Output video height." )#define RATIO_TEXT N_("Sample aspect ratio")#define RATIO_LONGTEXT N_( \    "Sample aspect ratio of the destination (1:1, 3:4, 2:3)." )#define VFILTER_TEXT N_("Video filter")#define VFILTER_LONGTEXT N_( \    "Video filters will be applied to the video stream." )#define CHROMA_TEXT N_("Image chroma")#define CHROMA_LONGTEXT N_( \    "Force the use of a specific chroma. Use YUVA if you're planning " \    "to use the Alphamask or Bluescreen video filter." )#define ALPHA_TEXT N_("Transparency")#define ALPHA_LONGTEXT N_( \    "Transparency of the mosaic picture." )#define X_TEXT N_("X offset")#define X_LONGTEXT N_( \    "X coordinate of the upper left corner in the mosaic if non negative." )#define Y_TEXT N_("Y offset")#define Y_LONGTEXT N_( \    "Y coordinate of the upper left corner in the mosaic if non negative." )#define CFG_PREFIX "sout-mosaic-bridge-"vlc_module_begin();    set_shortname( N_( "Mosaic bridge" ) );    set_description(N_("Mosaic bridge stream output") );    set_capability( "sout stream", 0 );    add_shortcut( "mosaic-bridge" );    add_string( CFG_PREFIX "id", "Id", NULL, ID_TEXT, ID_LONGTEXT,                false );    add_integer( CFG_PREFIX "width", 0, NULL, WIDTH_TEXT,                 WIDTH_LONGTEXT, true );    add_integer( CFG_PREFIX "height", 0, NULL, HEIGHT_TEXT,                 HEIGHT_LONGTEXT, true );    add_string( CFG_PREFIX "sar", "1:1", NULL, RATIO_TEXT,                RATIO_LONGTEXT, false );    add_string( CFG_PREFIX "chroma", 0, NULL, CHROMA_TEXT, CHROMA_LONGTEXT,                false );    add_module_list( CFG_PREFIX "vfilter", "video filter2",                     NULL, NULL, VFILTER_TEXT, VFILTER_LONGTEXT, false );    add_integer_with_range( CFG_PREFIX "alpha", 255, 0, 255, NULL,                            ALPHA_TEXT, ALPHA_LONGTEXT, false );    add_integer( CFG_PREFIX "x", -1, NULL, X_TEXT, X_LONGTEXT, false );    add_integer( CFG_PREFIX "y", -1, NULL, Y_TEXT, Y_LONGTEXT, false );    set_callbacks( Open, Close );vlc_module_end();static const char *const ppsz_sout_options[] = {    "id", "width", "height", "sar", "vfilter", "chroma", "alpha", "x", "y", NULL};/***************************************************************************** * Open *****************************************************************************/static int Open( vlc_object_t *p_this ){    sout_stream_t        *p_stream = (sout_stream_t *)p_this;    sout_stream_sys_t    *p_sys;    vlc_object_t         *p_libvlc = VLC_OBJECT( p_this->p_libvlc );    vlc_value_t           val;    config_ChainParse( p_stream, CFG_PREFIX, ppsz_sout_options,                       p_stream->p_cfg );    p_sys = malloc( sizeof( sout_stream_sys_t ) );    if( !p_sys )        return VLC_ENOMEM;    p_stream->p_sys = p_sys;    p_sys->b_inited = false;    var_Create( p_libvlc, "mosaic-lock", VLC_VAR_MUTEX );    var_Get( p_libvlc, "mosaic-lock", &val );    p_sys->p_lock = val.p_address;    p_sys->psz_id = var_CreateGetString( p_stream, CFG_PREFIX "id" );    p_sys->i_height =        var_CreateGetIntegerCommand( p_stream, CFG_PREFIX "height" );    var_AddCallback( p_stream, CFG_PREFIX "height", HeightCallback, p_stream );    p_sys->i_width =        var_CreateGetIntegerCommand( p_stream, CFG_PREFIX "width" );    var_AddCallback( p_stream, CFG_PREFIX "width", WidthCallback, p_stream );    var_Get( p_stream, CFG_PREFIX "sar", &val );    if( val.psz_string )    {        char *psz_parser = strchr( val.psz_string, ':' );        if( psz_parser )        {            *psz_parser++ = '\0';            p_sys->i_sar_num = atoi( val.psz_string );            p_sys->i_sar_den = atoi( psz_parser );            vlc_ureduce( &p_sys->i_sar_num, &p_sys->i_sar_den,                         p_sys->i_sar_num, p_sys->i_sar_den, 0 );        }        else        {            msg_Warn( p_stream, "bad aspect ratio %s", val.psz_string );            p_sys->i_sar_num = p_sys->i_sar_den = 1;        }        free( val.psz_string );    }    else    {        p_sys->i_sar_num = p_sys->i_sar_den = 1;    }    p_sys->i_chroma = 0;    val.psz_string = var_GetNonEmptyString( p_stream, CFG_PREFIX "chroma" );    if( val.psz_string && strlen( val.psz_string ) >= 4 )    {        memcpy( &p_sys->i_chroma, val.psz_string, 4 );        msg_Dbg( p_stream, "Forcing image chroma to 0x%.8x (%4.4s)", p_sys->i_chroma, (char*)&p_sys->i_chroma );    }    free( val.psz_string );#define INT_COMMAND( a ) do { \    var_Create( p_stream, CFG_PREFIX #a, \                VLC_VAR_INTEGER | VLC_VAR_DOINHERIT | VLC_VAR_ISCOMMAND ); \    var_AddCallback( p_stream, CFG_PREFIX #a, a ## Callback, \                     p_stream ); } while(0)    INT_COMMAND( alpha );    INT_COMMAND( x );    INT_COMMAND( y );#undef INT_COMMAND    p_stream->pf_add    = Add;    p_stream->pf_del    = Del;    p_stream->pf_send   = Send;    p_stream->p_sout->i_out_pace_nocontrol++;    return VLC_SUCCESS;}/***************************************************************************** * Close *****************************************************************************/static void Close( vlc_object_t * p_this ){    sout_stream_t     *p_stream = (sout_stream_t*)p_this;    sout_stream_sys_t *p_sys = p_stream->p_sys;    p_stream->p_sout->i_out_pace_nocontrol--;    free( p_sys->psz_id );    free( p_sys );}static int video_filter_buffer_allocation_init( filter_t *p_filter, void *p_data ){    p_filter->pf_vout_buffer_new = video_new_buffer_filter;    p_filter->pf_vout_buffer_del = video_del_buffer_filter;    p_filter->p_owner = p_data;    return VLC_SUCCESS;}static sout_stream_id_t * Add( sout_stream_t *p_stream, es_format_t *p_fmt ){    sout_stream_sys_t *p_sys = p_stream->p_sys;    bridge_t *p_bridge;    bridged_es_t *p_es;    char *psz_chain;    int i;    if( p_sys->b_inited || p_fmt->i_cat != VIDEO_ES )        return NULL;    /* Create decoder object */    p_sys->p_decoder = vlc_object_create( p_stream, VLC_OBJECT_DECODER );    if( !p_sys->p_decoder )        return NULL;    vlc_object_attach( p_sys->p_decoder, p_stream );    p_sys->p_decoder->p_module = NULL;    p_sys->p_decoder->fmt_in = *p_fmt;    p_sys->p_decoder->b_pace_control = false;    p_sys->p_decoder->fmt_out = p_sys->p_decoder->fmt_in;    p_sys->p_decoder->fmt_out.i_extra = 0;    p_sys->p_decoder->fmt_out.p_extra = 0;    p_sys->p_decoder->pf_decode_video = 0;    p_sys->p_decoder->pf_vout_buffer_new = video_new_buffer_decoder;    p_sys->p_decoder->pf_vout_buffer_del = video_del_buffer_decoder;    p_sys->p_decoder->pf_picture_link    = video_link_picture_decoder;    p_sys->p_decoder->pf_picture_unlink  = video_unlink_picture_decoder;    p_sys->p_decoder->p_owner = malloc( sizeof(decoder_owner_sys_t) );    if( !p_sys->p_decoder->p_owner )    {        vlc_object_detach( p_sys->p_decoder );        vlc_object_release( p_sys->p_decoder );        return NULL;    }    for( i = 0; i < PICTURE_RING_SIZE; i++ )        p_sys->p_decoder->p_owner->pp_pics[i] = NULL;    p_sys->p_decoder->p_owner->video = p_fmt->video;    //p_sys->p_decoder->p_cfg = p_sys->p_video_cfg;    p_sys->p_decoder->p_module =        module_Need( p_sys->p_decoder, "decoder", "$codec", 0 );    if( !p_sys->p_decoder->p_module || !p_sys->p_decoder->pf_decode_video )    {        if( p_sys->p_decoder->p_module )        {            msg_Err( p_stream, "instanciated a non video decoder" );            module_Unneed( p_sys->p_decoder, p_sys->p_decoder->p_module );        }        else        {            msg_Err( p_stream, "cannot find decoder" );        }        free( p_sys->p_decoder->p_owner );        vlc_object_detach( p_sys->p_decoder );        vlc_object_release( p_sys->p_decoder );        return NULL;    }    p_sys->b_inited = true;    vlc_mutex_lock( p_sys->p_lock );    p_bridge = GetBridge( p_stream );    if ( p_bridge == NULL )    {        vlc_object_t *p_libvlc = VLC_OBJECT( p_stream->p_libvlc );        vlc_value_t val;        p_bridge = malloc( sizeof( bridge_t ) );        var_Create( p_libvlc, "mosaic-struct", VLC_VAR_ADDRESS );        val.p_address = p_bridge;        var_Set( p_libvlc, "mosaic-struct", val );        p_bridge->i_es_num = 0;        p_bridge->pp_es = NULL;    }    for ( i = 0; i < p_bridge->i_es_num; i++ )    {        if ( p_bridge->pp_es[i]->b_empty )            break;    }    if ( i == p_bridge->i_es_num )    {        p_bridge->pp_es = realloc( p_bridge->pp_es,                                   (p_bridge->i_es_num + 1)                                     * sizeof(bridged_es_t *) );        p_bridge->i_es_num++;        p_bridge->pp_es[i] = malloc( sizeof(bridged_es_t) );    }    p_sys->p_es = p_es = p_bridge->pp_es[i];    p_es->i_alpha = var_GetInteger( p_stream, CFG_PREFIX "alpha" );    p_es->i_x = var_GetInteger( p_stream, CFG_PREFIX "x" );    p_es->i_y = var_GetInteger( p_stream, CFG_PREFIX "y" );    //p_es->fmt = *p_fmt;    p_es->psz_id = p_sys->psz_id;    p_es->p_picture = NULL;    p_es->pp_last = &p_es->p_picture;    p_es->b_empty = false;    vlc_mutex_unlock( p_sys->p_lock );    if ( p_sys->i_height || p_sys->i_width )    {        p_sys->p_image = image_HandlerCreate( p_stream );    }    else    {        p_sys->p_image = NULL;    }    msg_Dbg( p_stream, "mosaic bridge id=%s pos=%d", p_es->psz_id, i );    /* Create user specified video filters */    psz_chain = var_GetNonEmptyString( p_stream, CFG_PREFIX "vfilter" );    msg_Dbg( p_stream, "psz_chain: %s\n", psz_chain );    if( psz_chain )    {        p_sys->p_vf2 = filter_chain_New( p_stream, "video filter2", false,                                         video_filter_buffer_allocation_init,                                         NULL, p_sys->p_decoder->p_owner );        es_format_t fmt;        es_format_Copy( &fmt, &p_sys->p_decoder->fmt_out );        if( p_sys->i_chroma )            fmt.video.i_chroma = p_sys->i_chroma;        filter_chain_Reset( p_sys->p_vf2, &fmt, &fmt );        filter_chain_AppendFromString( p_sys->p_vf2, psz_chain );        free( psz_chain );    }    else    {        p_sys->p_vf2 = NULL;    }    return (sout_stream_id_t *)p_sys;}static int Del( sout_stream_t *p_stream, sout_stream_id_t *id ){    VLC_UNUSED(id);    sout_stream_sys_t *p_sys = p_stream->p_sys;    bridge_t *p_bridge;    bridged_es_t *p_es;    bool b_last_es = true;    int i;    if( !p_sys->b_inited )        return VLC_SUCCESS;    if( p_sys->p_decoder != NULL )    {        decoder_owner_sys_t *p_owner = p_sys->p_decoder->p_owner;

⌨️ 快捷键说明

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