📄 timeshift.c
字号:
/***************************************************************************** * timeshift.c: access filter implementing timeshifting capabilities ***************************************************************************** * Copyright (C) 2005 the VideoLAN team * $Id: timeshift.c 16434 2006-08-30 15:18:13Z hartman $ * * Authors: Laurent Aimar <fenrir@via.ecp.fr> * Gildas Bazin <gbazin@videolan.org> * * 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 *****************************************************************************/#include <stdlib.h>#include <errno.h>#include <vlc/vlc.h>#include <vlc/input.h>#include "charset.h"#include <unistd.h>/***************************************************************************** * Module descriptor *****************************************************************************/static int Open ( vlc_object_t * );static void Close( vlc_object_t * );#define GRANULARITY_TEXT N_("Timeshift granularity")/// \bug [String] typo#define GRANULARITY_LONGTEXT N_( "This is the size of the temporary files " \ "tha will be used to store the timeshifted streams." )#define DIR_TEXT N_("Timeshift directory")#define DIR_LONGTEXT N_( "Directory used to store the timeshift temporary " \ "files." )#define FORCE_TEXT N_("Force use of the timeshift module")#define FORCE_LONGTEXT N_("Force use of the timeshift module even if the " \ "access declares that it can control pace or pause." )vlc_module_begin(); set_shortname( _("Timeshift") ); set_description( _("Timeshift") ); set_category( CAT_INPUT ); set_subcategory( SUBCAT_INPUT_ACCESS_FILTER ); set_capability( "access_filter", 0 ); add_shortcut( "timeshift" ); set_callbacks( Open, Close ); add_integer( "timeshift-granularity", 50, NULL, GRANULARITY_TEXT, GRANULARITY_LONGTEXT, VLC_TRUE ); add_directory( "timeshift-dir", 0, 0, DIR_TEXT, DIR_LONGTEXT, VLC_FALSE ); add_bool( "timeshift-force", VLC_FALSE, NULL, FORCE_TEXT, FORCE_LONGTEXT, VLC_FALSE );vlc_module_end();/***************************************************************************** * Local prototypes *****************************************************************************/static int Seek( access_t *, int64_t );static block_t *Block ( access_t *p_access );static int Control( access_t *, int i_query, va_list args );static void Thread ( access_t *p_access );static int WriteBlockToFile( access_t *p_access, block_t *p_block );static block_t *ReadBlockFromFile( access_t *p_access );static char *GetTmpFilePath( access_t *p_access );#define TIMESHIFT_FIFO_MAX (10*1024*1024)#define TIMESHIFT_FIFO_MIN (TIMESHIFT_FIFO_MAX/4)#define TMP_FILE_MAX 256typedef struct ts_entry_t{ FILE *file; struct ts_entry_t *p_next;} ts_entry_t;struct access_sys_t{ block_fifo_t *p_fifo; int i_files; int i_file_size; int i_write_size; ts_entry_t *p_read_list; ts_entry_t **pp_read_last; ts_entry_t *p_write_list; ts_entry_t **pp_write_last; char *psz_filename_base; char *psz_filename;};/***************************************************************************** * 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; vlc_bool_t b_bool; var_Create( p_access, "timeshift-force", VLC_VAR_BOOL | VLC_VAR_DOINHERIT ); if( var_GetBool( p_access, "timeshift-force" ) == VLC_TRUE ) { msg_Dbg( p_access, "Forcing use of timeshift even if access can control pace or pause" ); } else { /* Only work with not pace controled access */ if( access2_Control( p_src, ACCESS_CAN_CONTROL_PACE, &b_bool ) || b_bool ) { msg_Dbg( p_src, "ACCESS_CAN_CONTROL_PACE: timeshift useless" ); return VLC_EGENERIC; } /* Refuse access that can be paused */ if( access2_Control( p_src, ACCESS_CAN_PAUSE, &b_bool ) || b_bool ) { msg_Dbg( p_src, "ACCESS_CAN_PAUSE: timeshift useless" ); return VLC_EGENERIC; } } /* */ p_access->pf_read = NULL; p_access->pf_block = Block; p_access->pf_seek = Seek; p_access->pf_control = Control; p_access->info = p_src->info; p_access->p_sys = p_sys = malloc( sizeof( access_sys_t ) ); /* */ p_sys->p_fifo = block_FifoNew( p_access ); p_sys->i_write_size = 0; p_sys->i_files = 0; p_sys->p_read_list = NULL; p_sys->pp_read_last = &p_sys->p_read_list; p_sys->p_write_list = NULL; p_sys->pp_write_last = &p_sys->p_write_list; var_Create( p_access, "timeshift-dir", VLC_VAR_DIRECTORY | VLC_VAR_DOINHERIT ); var_Create( p_access, "timeshift-granularity", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT ); p_sys->i_file_size = var_GetInteger( p_access, "timeshift-granularity" ); if( p_sys->i_file_size < 1 ) p_sys->i_file_size = 1; p_sys->i_file_size *= 1024 * 1024; /* In MBytes */ p_sys->psz_filename_base = GetTmpFilePath( p_access ); p_sys->psz_filename = malloc( strlen( p_sys->psz_filename_base ) + 1000 ); if( vlc_thread_create( p_access, "timeshift thread", Thread, VLC_THREAD_PRIORITY_LOW, VLC_FALSE ) ) { msg_Err( p_access, "cannot spawn timeshift access thread" ); return VLC_EGENERIC; } 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; ts_entry_t *p_entry; int i; msg_Dbg( p_access, "timeshift close called" ); vlc_thread_join( p_access ); for( p_entry = p_sys->p_write_list; p_entry; ) { ts_entry_t *p_next = p_entry->p_next; fclose( p_entry->file ); free( p_entry ); p_entry = p_next; } for( p_entry = p_sys->p_read_list; p_entry; ) { ts_entry_t *p_next = p_entry->p_next; fclose( p_entry->file ); free( p_entry ); p_entry = p_next; } for( i = 0; i < p_sys->i_files; i++ ) { sprintf( p_sys->psz_filename, "%s%i.dat", p_sys->psz_filename_base, i ); unlink( p_sys->psz_filename ); } free( p_sys->psz_filename ); free( p_sys->psz_filename_base ); block_FifoRelease( p_sys->p_fifo ); free( p_sys );}/***************************************************************************** * *****************************************************************************/static block_t *Block( access_t *p_access ){ access_sys_t *p_sys = p_access->p_sys; block_t *p_block; if( p_access->b_die ) { p_access->info.b_eof = VLC_TRUE; return NULL; } p_block = block_FifoGet( p_sys->p_fifo ); //p_access->info.i_size -= p_block->i_buffer; return p_block;}/***************************************************************************** * *****************************************************************************/static void Thread( access_t *p_access ){ access_sys_t *p_sys = p_access->p_sys; access_t *p_src = p_access->p_source; int i; while( !p_access->b_die ) { block_t *p_block; /* Get a new block from the source */ if( p_src->pf_block ) { p_block = p_src->pf_block( p_src ); if( p_block == NULL ) { if( p_src->info.b_eof ) break; msleep( 1000 ); continue; } } else { if( ( p_block = block_New( p_access, 2048 ) ) == NULL ) break; p_block->i_buffer = p_src->pf_read( p_src, p_block->p_buffer, 2048 ); if( p_block->i_buffer < 0 ) { block_Release( p_block ); if( p_block->i_buffer == 0 ) break; msleep( 1000 ); continue; } } /* Write block */ if( !p_sys->p_write_list && !p_sys->p_read_list && p_sys->p_fifo->i_size < TIMESHIFT_FIFO_MAX ) { /* If there isn't too much timeshifted data, * write directly to FIFO */ block_FifoPut( p_sys->p_fifo, p_block ); //p_access->info.i_size += p_block->i_buffer; //p_access->info.i_update |= INPUT_UPDATE_SIZE; /* Nothing else to do */ continue;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -