]> git.sesse.net Git - vlc/commitdiff
* all: added a new access_filter module: record + one hotkey to start/stop
authorLaurent Aimar <fenrir@videolan.org>
Tue, 26 Apr 2005 07:28:07 +0000 (07:28 +0000)
committerLaurent Aimar <fenrir@videolan.org>
Tue, 26 Apr 2005 07:28:07 +0000 (07:28 +0000)
 recording (simple dumping). (Usefull for dvb/udp/mp3 radio stream).

include/vlc_keys.h
modules/access_filter/Modules.am
modules/access_filter/record.c [new file with mode: 0644]
src/libvlc.h

index 987d4ebfe5f7ca910a5079d0fd68b52461590d57..d872a746c22c15f65b5485ee922966ca32b17328 100644 (file)
@@ -242,4 +242,4 @@ static inline int StringToKey( char *psz_key )
 #define ACTIONID_AUDIODELAY_UP         53
 #define ACTIONID_AUDIODELAY_DOWN       54
 #define ACTIONID_SNAPSHOT              55
-
+#define ACTIONID_RECORD                56
index 88e20b934cd4719bc2bd6ccc006e4f93a6d33108..016ddcc42eae1e2ec9b4359ca8d68280697da381 100755 (executable)
@@ -1 +1,2 @@
 SOURCES_access_filter_timeshift = timeshift.c
+SOURCES_access_filter_record = record.c
diff --git a/modules/access_filter/record.c b/modules/access_filter/record.c
new file mode 100644 (file)
index 0000000..757c7da
--- /dev/null
@@ -0,0 +1,373 @@
+/*****************************************************************************
+ * record.c
+ *****************************************************************************
+ * Copyright (C) 2005 VideoLAN
+ * $Id: demux.c 7546 2004-04-29 13:53:29Z gbazin $
+ *
+ * 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., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
+ *****************************************************************************/
+
+/*****************************************************************************
+ * Preamble
+ *****************************************************************************/
+#include <stdlib.h>
+
+#include <vlc/vlc.h>
+#include <vlc/input.h>
+#include <vlc/vout.h>
+
+#include "vlc_keys.h"
+#include <osd.h>
+
+/*****************************************************************************
+ * Module descriptor
+ *****************************************************************************/
+
+#define RECORD_PATH_TXT N_("Record directory")
+#define RECORD_PATH_LONGTXT N_( \
+    "Allows you to specify the directory where the record will be stored" )
+
+static int  Open ( vlc_object_t * );
+static void Close( vlc_object_t * );
+
+vlc_module_begin();
+    set_description( _("Record") );
+    set_category( CAT_INPUT );
+    set_subcategory( SUBCAT_INPUT_ACCESS_FILTER );
+    set_capability( "access_filter", 0 );
+    add_shortcut( "record" );
+
+    add_string( "record-path", NULL, NULL,
+                RECORD_PATH_TXT, RECORD_PATH_LONGTXT, VLC_TRUE );
+
+    set_callbacks( Open, Close );
+
+vlc_module_end();
+
+/*****************************************************************************
+ * Local prototypes
+ *****************************************************************************/
+
+static block_t *Block  ( access_t * );
+static int      Read   ( access_t *, uint8_t *, int );
+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 * );
+
+struct access_sys_t
+{
+    vlc_bool_t b_dump;
+
+    char *psz_path;
+    char *psz_ext;
+    char *psz_file;
+    int64_t i_size;
+    FILE *f;
+
+    vout_thread_t *p_vout;
+    int            i_vout_chan;
+};
+
+/*****************************************************************************
+ * 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 ) );
+
+    /* */
+    p_sys->f = NULL;
+    p_sys->i_size = 0;
+    p_sys->psz_file = NULL;
+    p_sys->psz_ext = "dat";
+    p_sys->b_dump = VLC_FALSE;
+    p_sys->p_vout = NULL;
+    p_sys->i_vout_chan = -1;
+
+    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 );
+        if( p_access->p_vlc->psz_homedir )
+            psz = strdup( p_access->p_vlc->psz_homedir );
+    }
+    p_sys->psz_path = psz;
+    msg_Dbg( p_access, "Record access filter path %s", psz );
+
+    /* catch all key event */
+    var_AddCallback( p_access->p_vlc, "key-pressed", 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_vlc, "key-pressed", EventKey, p_access );
+
+    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;
+
+    /* */
+    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 );
+
+    /* */
+    p_access->info = p_src->info;
+
+    return p_block;
+}
+
+/*****************************************************************************
+ *
+ *****************************************************************************/
+static int Read( access_t *p_access, uint8_t *p_buffer, int i_len )
+{
+    access_t     *p_src = p_access->p_source;
+    int i_ret;
+
+    i_ret = p_src->pf_read( p_src, p_buffer, i_len );
+
+    if( i_ret > 0 )
+        Dump( p_access, p_buffer, i_ret );
+
+    /* */
+    p_access->info = p_src->info;
+
+    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;
+
+    i_ret = p_src->pf_control( p_src, i_query, args );
+
+    /* */
+    p_access->info = p_src->info;
+
+    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;
+
+    i_ret = p_src->pf_seek( p_src, i_pos );
+
+    /* */
+    p_access->info = p_src->info;
+
+    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;
+
+    struct hotkey *p_hotkeys = p_access->p_vlc->p_hotkeys;
+    int i_action = -1, i;
+
+    for( i = 0; p_hotkeys[i].psz_action != NULL; i++ )
+    {
+        if( p_hotkeys[i].i_key == newval.i_int )
+        {
+            i_action = p_hotkeys[i].i_action;
+        }
+    }
+
+    if( i_action == ACTIONID_RECORD )
+    {
+        if( p_sys->b_dump )
+            p_sys->b_dump = VLC_FALSE;
+        else
+            p_sys->b_dump = VLC_TRUE;
+    }
+
+    return VLC_SUCCESS;
+}
+
+/*****************************************************************************
+ *
+ *****************************************************************************/
+static void Notify( access_t *p_access, vlc_bool_t 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 != 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 "I64Fd" kb (%s)",
+                     p_sys->i_size/1024, p_sys->psz_file );
+
+            Notify( p_access, VLC_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;
+        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 )
+        {
+            vlc_mutex_lock( &p_input->input.p_item->lock );
+            if( p_input->input.p_item->psz_name &&
+                strlen( p_input->input.p_item->psz_name ) < 64 )
+                psz_name = strdup( p_input->input.p_item->psz_name );
+            vlc_mutex_unlock( &p_input->input.p_item->lock );
+
+            vlc_object_release( p_input );
+        }
+        if( psz_name == NULL )
+            psz_name = strdup( "Unknown" );
+
+        asprintf( &p_sys->psz_file, "%s/%s %d-%d-%d %.2dh%.2dm%.2ds.%s",
+                  p_sys->psz_path, psz_name,
+                  l.tm_mday, l.tm_mon+1, l.tm_year+1900,
+                  l.tm_hour, l.tm_min, l.tm_sec,
+                  p_sys->psz_ext );
+
+        free( psz_name );
+
+        msg_Dbg( p_access, "dump in file '%s'", p_sys->psz_file );
+
+        p_sys->f = fopen( p_sys->psz_file, "wb" );
+        if( p_sys->f == NULL )
+        {
+            msg_Err( p_access, "cannot open file '%s' (%s)",
+                     p_sys->psz_file, strerror(errno) );
+            free( p_sys->psz_file );
+            p_sys->psz_file = NULL;
+            p_sys->b_dump = VLC_FALSE;
+            return;
+        }
+
+        Notify( p_access, VLC_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;
+}
+
index a93ba378e97d31893401a51e462bcb808777f649..a928cd4c67c568dc7a67e77770ba59a1dfc26919 100644 (file)
@@ -846,6 +846,10 @@ static char *ppsz_clock_descriptions[] =
 #define SNAP_KEY_TEXT N_("Take video snapshot")
 #define SNAP_KEY_LONGTEXT N_("Takes a video snapshot and writes it to disk.")
 
+#define RECORD_KEY_TEXT N_("Record")
+#define RECORD_KEY_LONGTEXT N_("Record access filter start/stop.")
+
+
 #define VLC_USAGE N_( \
     "Usage: %s [options] [playlistitems] ..." \
     "\nYou can specify multiple playlistitems on the commandline. They will be enqueued in the playlist." \
@@ -1363,6 +1367,7 @@ vlc_module_begin();
 #   define KEY_PLAY_BOOKMARK10    KEY_UNSET
 #   define KEY_HISTORY_BACK       KEY_MODIFIER_COMMAND|'['
 #   define KEY_HISTORY_FORWARD    KEY_MODIFIER_COMMAND|']'
+#   define KEY_RECORD             KEY_UNSET
 
 #else
 #   define KEY_FULLSCREEN         'f'
@@ -1422,6 +1427,7 @@ vlc_module_begin();
 #   define KEY_PLAY_BOOKMARK10    KEY_F10
 #   define KEY_HISTORY_BACK       KEY_MODIFIER_CTRL|'['
 #   define KEY_HISTORY_FORWARD    KEY_MODIFIER_CTRL|']'
+#   define KEY_RECORD             KEY_MODIFIER_CTRL|'r'
 #endif
 
     add_key( "key-fullscreen", KEY_FULLSCREEN, NULL, FULLSCREEN_KEY_TEXT,
@@ -1537,6 +1543,8 @@ vlc_module_begin();
              HISTORY_BACK_LONGTEXT, VLC_TRUE );
     add_key( "key-history-forward", KEY_HISTORY_FORWARD, NULL,
              HISTORY_FORWARD_TEXT, HISTORY_FORWARD_LONGTEXT, VLC_TRUE );
+    add_key( "key-record", KEY_RECORD, NULL,
+             RECORD_KEY_TEXT, RECORD_KEY_LONGTEXT, VLC_TRUE );
 
     /* Usage (mainly useful for cmd line stuff) */
     /* add_usage_hint( PLAYLIST_USAGE ); */
@@ -1636,6 +1644,7 @@ static struct hotkey p_hotkeys[] =
     { "key-play-bookmark10", ACTIONID_PLAY_BOOKMARK10, 0},
     { "key-history-back", ACTIONID_HISTORY_BACK, 0},
     { "key-history-forward", ACTIONID_HISTORY_FORWARD, 0},
+    { "key-record", ACTIONID_RECORD, 0 },
     { NULL, 0, 0 }
 };