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

📄 record.c

📁 VLC Player Source Code
💻 C
字号:
/***************************************************************************** * record.c ***************************************************************************** * Copyright (C) 2005-2006 the VideoLAN team * $Id: 67dba9d280a221898da2188881f4ac723b6a6ddf $ * * Author: Laurent Aimar <fenrir@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 <vlc_common.h>#include <vlc_plugin.h>#include <vlc_input.h>#include <vlc_access.h>#include "vlc_keys.h"#include <vlc_osd.h>#include <vlc_charset.h>#include <errno.h>#include <time.h>/***************************************************************************** * Module descriptor *****************************************************************************/#define RECORD_PATH_TXT N_("Record directory")#define RECORD_PATH_LONGTXT N_( \    "Directory where the record will be stored." )static int  Open ( vlc_object_t * );static void Close( vlc_object_t * );vlc_module_begin();    set_shortname( N_("Record") );    set_description( N_("Record") );    set_category( CAT_INPUT );    set_subcategory( SUBCAT_INPUT_ACCESS_FILTER );    set_capability( "access_filter", 0 );    add_shortcut( "record" );    add_directory( "record-path", NULL, NULL,                   RECORD_PATH_TXT, RECORD_PATH_LONGTXT, true );        change_unsafe();    set_callbacks( Open, Close );vlc_module_end();/***************************************************************************** * Local prototypes *****************************************************************************/static block_t *Block  ( access_t * );static ssize_t  Read   ( access_t *, uint8_t *, size_t );static int      Control( access_t *, int i_query, va_list args );static int      Seek   ( access_t *, int64_t );static void Dump( access_t *, uint8_t *, int );static int EventKey( vlc_object_t *, char const *,                     vlc_value_t, vlc_value_t, void * );static int ToggleRecord( vlc_object_t *, char const *,                         vlc_value_t, vlc_value_t, void * );struct access_sys_t{    bool b_dump;    char *psz_path;    const char *psz_ext;    char *psz_file;    int64_t i_size;    FILE *f;    vout_thread_t *p_vout;    int            i_vout_chan;    int i_update_sav;};static inline void PreUpdateFlags( access_t *p_access ){    access_t *p_src = p_access->p_source;    /* backport flags turned off 0 */    p_src->info.i_update &= p_access->p_sys->i_update_sav ^ (~p_access->info.i_update);}static inline void PostUpdateFlags( access_t *p_access ){    access_t *p_src = p_access->p_source;    p_access->info = p_src->info;    p_access->p_sys->i_update_sav = p_access->info.i_update;}/***************************************************************************** * Open: *****************************************************************************/static int Open( vlc_object_t *p_this ){    access_t *p_access = (access_t*)p_this;    access_t *p_src = p_access->p_source;    access_sys_t *p_sys;    char *psz;    p_access->pf_read  = p_src->pf_read  ? Read : NULL;    p_access->pf_block = p_src->pf_block ? Block : NULL;    p_access->pf_seek  = p_src->pf_seek  ? Seek : NULL;    p_access->pf_control = Control;    p_access->info = p_src->info;    p_access->p_sys = p_sys = malloc( sizeof( access_t ) );    if( !p_sys ) return VLC_ENOMEM;    /* */    p_sys->f = NULL;    p_sys->i_size = 0;    p_sys->psz_file = NULL;    p_sys->psz_ext = "dat";    p_sys->b_dump = false;    p_sys->p_vout = NULL;    p_sys->i_vout_chan = -1;    p_sys->i_update_sav = p_access->info.i_update;    if( !strncasecmp( p_src->psz_access, "dvb", 3 ) ||        !strncasecmp( p_src->psz_access, "udp", 3 ) )        p_sys->psz_ext = "ts";    psz = var_CreateGetString( p_access, "record-path" );    if( *psz == '\0' )    {        free( psz );        psz = strdup( config_GetHomeDir() );    }    p_sys->psz_path = psz;    msg_Dbg( p_access, "Record access filter path %s", psz );    input_thread_t *p_input = ( input_thread_t * )            vlc_object_find( p_access, VLC_OBJECT_INPUT, FIND_PARENT );    if( p_input )    {        var_Create( p_input, "record-toggle", VLC_VAR_VOID );        var_AddCallback( p_input, "record-toggle", ToggleRecord, p_access );        vlc_object_release( p_input );    }    /* catch all key event */    var_AddCallback( p_access->p_libvlc, "key-action", EventKey, p_access );    return VLC_SUCCESS;}/***************************************************************************** * Close: *****************************************************************************/static void Close( vlc_object_t *p_this ){    access_t     *p_access = (access_t*)p_this;    access_sys_t *p_sys = p_access->p_sys;    var_DelCallback( p_access->p_libvlc, "key-action", EventKey, p_access );    input_thread_t *p_input = ( input_thread_t * )            vlc_object_find( p_access, VLC_OBJECT_INPUT, FIND_PARENT );    if( p_input )    {        var_Destroy( p_input, "record-toggle" );        vlc_object_release( p_input );    }    if( p_sys->f )    {        fclose( p_sys->f );        free( p_sys->psz_file );    }    free( p_sys->psz_path );    free( p_sys );}/***************************************************************************** * *****************************************************************************/static block_t *Block( access_t *p_access ){    access_t     *p_src = p_access->p_source;    block_t      *p_block;    PreUpdateFlags( p_access );    p_block = p_src->pf_block( p_src );    if( p_block && p_block->i_buffer )        Dump( p_access, p_block->p_buffer, p_block->i_buffer );    PostUpdateFlags( p_access );    return p_block;}/***************************************************************************** * *****************************************************************************/static ssize_t Read( access_t *p_access, uint8_t *p_buffer, size_t i_len ){    access_t     *p_src = p_access->p_source;    int i_ret;    PreUpdateFlags( p_access );    i_ret = p_src->pf_read( p_src, p_buffer, i_len );    if( i_ret > 0 )        Dump( p_access, p_buffer, i_ret );    PostUpdateFlags( p_access );    return i_ret;}/***************************************************************************** * *****************************************************************************/static int Control( access_t *p_access, int i_query, va_list args ){    access_t     *p_src = p_access->p_source;    int i_ret;    PreUpdateFlags( p_access );    i_ret = p_src->pf_control( p_src, i_query, args );    PostUpdateFlags( p_access );    return i_ret;}/***************************************************************************** * *****************************************************************************/static int Seek( access_t *p_access, int64_t i_pos ){    access_t     *p_src = p_access->p_source;    int i_ret;    PreUpdateFlags( p_access );    i_ret = p_src->pf_seek( p_src, i_pos );    PostUpdateFlags( p_access );    return i_ret;}/***************************************************************************** * *****************************************************************************/static int EventKey( vlc_object_t *p_this, char const *psz_var,                     vlc_value_t oldval, vlc_value_t newval, void *p_data ){    access_t     *p_access = p_data;    access_sys_t *p_sys = p_access->p_sys;    (void)p_this;    (void)psz_var;    (void)oldval;    if( newval.i_int == ACTIONID_RECORD )    {        if( p_sys->b_dump )            p_sys->b_dump = false;        else            p_sys->b_dump = true;    }    return VLC_SUCCESS;}static int ToggleRecord( vlc_object_t *p_this, char const *psz_var,                          vlc_value_t oldval, vlc_value_t newval,                          void *p_data ){    access_t     *p_access = p_data;    access_sys_t *p_sys = p_access->p_sys;    (void)p_this;    (void)psz_var;    (void)oldval;    (void)newval;    p_sys->b_dump = !p_sys->b_dump;    return VLC_SUCCESS;}/***************************************************************************** * *****************************************************************************/static void Notify( access_t *p_access, bool b_dump ){    access_sys_t *p_sys = p_access->p_sys;    vout_thread_t *p_vout;    p_vout = vlc_object_find( p_access, VLC_OBJECT_VOUT, FIND_ANYWHERE );    if( !p_vout ) return;    if( p_vout != p_sys->p_vout )    {        p_sys->p_vout = p_vout;        if( spu_Control( p_vout->p_spu, SPU_CHANNEL_REGISTER,                         &p_sys->i_vout_chan  ) )            p_sys->i_vout_chan = -1;    }    if( p_sys->i_vout_chan != -1 )    {        if( b_dump )            vout_OSDMessage( p_vout, p_sys->i_vout_chan, _("Recording") );        else            vout_OSDMessage( p_vout, p_sys->i_vout_chan, _("Recording done") );    }    vlc_object_release( p_vout );}/***************************************************************************** * *****************************************************************************/static void Dump( access_t *p_access, uint8_t *p_buffer, int i_buffer ){    access_sys_t *p_sys = p_access->p_sys;    int i_write;    if( !p_sys->b_dump )    {        if( p_sys->f )        {            msg_Dbg( p_access, "dumped %"PRId64" kb (%s)",                     p_sys->i_size/1024, p_sys->psz_file );            Notify( p_access, false );            fclose( p_sys->f );            p_sys->f = NULL;            free( p_sys->psz_file );            p_sys->psz_file = NULL;            p_sys->i_size = 0;        }        return;    }    if( !p_sys->f )    {        input_thread_t *p_input;        char *psz_name = NULL, *psz;        time_t t = time(NULL);        struct tm l;        if( !localtime_r( &t, &l ) ) memset( &l, 0, sizeof(l) );        p_input = vlc_object_find( p_access, VLC_OBJECT_INPUT, FIND_PARENT );        if( p_input )        {            input_item_t * p_item = input_GetItem( p_input );            vlc_mutex_lock( &p_item->lock );            if( p_item->psz_name )            {                char *p = strrchr( p_item->psz_name, '/' );                if( p == NULL )                    p = strrchr( p_item->psz_name, '\\' );                if( p == NULL )                    psz_name = strdup( p_item->psz_name );                else if( p[1] != '\0' )                    psz_name = strdup( &p[1] );            }            vlc_mutex_unlock( &p_item->lock );            vlc_object_release( p_input );        }        if( asprintf( &p_sys->psz_file, "%s %d-%d-%d %.2dh%.2dm%.2ds.%s",                      ( psz_name != NULL ) ? psz_name : "Unknown",                      l.tm_mday, l.tm_mon+1, l.tm_year+1900,                      l.tm_hour, l.tm_min, l.tm_sec,                      p_sys->psz_ext ) == -1 )            p_sys->psz_file = NULL;        free( psz_name );        if( p_sys->psz_file == NULL )        {            p_sys->b_dump = false;            return;        }        /* Remove all forbidden characters (except (back)slashes) */        for( psz = p_sys->psz_file; *psz; psz++ )        {            unsigned char c = (unsigned char)*psz;            /* Even if many OS accept non printable characters, we remove             * them to avoid confusing users */            if( ( c < 32 ) || ( c == 127 ) )                *psz = '_';#if defined (WIN32) || defined (UNDER_CE)            /* Windows has a lot of forbidden characters, even if it has             * fewer than DOS. */            if( strchr( "\"*:<>?|", c ) != NULL )                *psz = '_';#endif        }        psz_name = p_sys->psz_file;#if defined (WIN32) || defined (UNDER_CE)#define DIR_SEP "\\"#else#define DIR_SEP "/"#endif        if( asprintf( &p_sys->psz_file, "%s" DIR_SEP "%s",                      p_sys->psz_path, psz_name ) == -1 )            p_sys->psz_file = NULL;        free( psz_name );        if( p_sys->psz_file == NULL )        {            p_sys->b_dump = false;            return;        }        msg_Dbg( p_access, "dump in file '%s'", p_sys->psz_file );        p_sys->f = utf8_fopen( p_sys->psz_file, "wb" );        if( p_sys->f == NULL )        {            msg_Err( p_access, "cannot open file '%s' (%m)",                     p_sys->psz_file );            free( p_sys->psz_file );            p_sys->psz_file = NULL;            p_sys->b_dump = false;            return;        }        Notify( p_access, true );        p_sys->i_size = 0;    }    if( ( i_write = fwrite( p_buffer, 1, i_buffer, p_sys->f ) ) > 0 )        p_sys->i_size += i_write;}

⌨️ 快捷键说明

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