record.c
上传用户:kjfoods
上传日期:2020-07-06
资源大小:29949k
文件大小:7k
源码类别:

midi

开发平台:

Unix_Linux

  1. /*****************************************************************************
  2.  * record.c
  3.  *****************************************************************************
  4.  * Copyright (C) 2008 Laurent Aimar
  5.  * $Id: 2ab138fd452b81f52ffb4bc8a26c515c69b5e905 $
  6.  *
  7.  * Author: Laurent Aimar <fenrir _AT_ videolan _DOT_ org>
  8.  *
  9.  * This program is free software; you can redistribute it and/or modify
  10.  * it under the terms of the GNU General Public License as published by
  11.  * the Free Software Foundation; either version 2 of the License, or
  12.  * (at your option) any later version.
  13.  *
  14.  * This program is distributed in the hope that it will be useful,
  15.  * but WITHOUT ANY WARRANTY; without even the implied warranty of
  16.  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  17.  * GNU General Public License for more details.
  18.  *
  19.  * You should have received a copy of the GNU General Public License
  20.  * along with this program; if not, write to the Free Software
  21.  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
  22.  *****************************************************************************/
  23. /*****************************************************************************
  24.  * Preamble
  25.  *****************************************************************************/
  26. #ifdef HAVE_CONFIG_H
  27. # include "config.h"
  28. #endif
  29. #include <vlc_common.h>
  30. #include <vlc_plugin.h>
  31. #include <assert.h>
  32. #include <vlc_stream.h>
  33. #include <vlc_input.h>
  34. #include <vlc_charset.h>
  35. /*****************************************************************************
  36.  * Module descriptor
  37.  *****************************************************************************/
  38. static int  Open ( vlc_object_t * );
  39. static void Close( vlc_object_t * );
  40. vlc_module_begin()
  41.     set_category( CAT_INPUT )
  42.     set_subcategory( SUBCAT_INPUT_STREAM_FILTER )
  43.     set_description( N_("Internal stream record") )
  44.     set_capability( "stream_filter", 0 )
  45.     set_callbacks( Open, Close )
  46. vlc_module_end()
  47. /*****************************************************************************
  48.  *
  49.  *****************************************************************************/
  50. struct stream_sys_t
  51. {
  52.     bool b_active;
  53.     FILE *f;        /* TODO it could be replaced by access_output_t one day */
  54.     bool b_error;
  55. };
  56. /****************************************************************************
  57.  * Local prototypes
  58.  ****************************************************************************/
  59. static int  Read   ( stream_t *, void *p_read, unsigned int i_read );
  60. static int  Peek   ( stream_t *, const uint8_t **pp_peek, unsigned int i_peek );
  61. static int  Control( stream_t *, int i_query, va_list );
  62. static int  Start  ( stream_t *, const char *psz_extension );
  63. static int  Stop   ( stream_t * );
  64. static void Write  ( stream_t *, const uint8_t *p_buffer, size_t i_buffer );
  65. /****************************************************************************
  66.  * Open
  67.  ****************************************************************************/
  68. static int Open ( vlc_object_t *p_this )
  69. {
  70.     stream_t *s = (stream_t*)p_this;
  71.     stream_sys_t *p_sys;
  72.     /* */
  73.     s->p_sys = p_sys = malloc( sizeof( *p_sys ) );
  74.     if( !p_sys )
  75.         return VLC_ENOMEM;
  76.     p_sys->b_active = false;
  77.     /* */
  78.     s->pf_read = Read;
  79.     s->pf_peek = Peek;
  80.     s->pf_control = Control;
  81.     return VLC_SUCCESS;
  82. }
  83. /****************************************************************************
  84.  * Close
  85.  ****************************************************************************/
  86. static void Close( vlc_object_t *p_this )
  87. {
  88.     stream_t *s = (stream_t*)p_this;
  89.     stream_sys_t *p_sys = s->p_sys;
  90.     if( p_sys->b_active )
  91.         Stop( s );
  92.     free( p_sys );
  93. }
  94. /****************************************************************************
  95.  * Stream filters functions
  96.  ****************************************************************************/
  97. static int Read( stream_t *s, void *p_read, unsigned int i_read )
  98. {
  99.     stream_sys_t *p_sys = s->p_sys;
  100.     void *p_record = p_read;
  101.     /* Allocate a temporary buffer for record when no p_read */
  102.     if( p_sys->b_active && !p_record )
  103.         p_record = malloc( i_read );
  104.     /* */
  105.     const int i_record = stream_Read( s->p_source, p_record, i_read );
  106.     /* Dump read data */
  107.     if( p_sys->b_active )
  108.     {
  109.         if( p_record && i_record > 0 )
  110.             Write( s, p_record, i_record );
  111.         if( !p_read )
  112.             free( p_record );
  113.     }
  114.     return i_record;
  115. }
  116. static int Peek( stream_t *s, const uint8_t **pp_peek, unsigned int i_peek )
  117. {
  118.     return stream_Peek( s->p_source, pp_peek, i_peek );
  119. }
  120. static int Control( stream_t *s, int i_query, va_list args )
  121. {
  122.     if( i_query != STREAM_SET_RECORD_STATE )
  123.         return stream_vaControl( s->p_source, i_query, args );
  124.     bool b_active = (bool)va_arg( args, int );
  125.     const char *psz_extension = NULL;
  126.     if( b_active )
  127.         psz_extension = (const char*)va_arg( args, const char* );
  128.     if( !s->p_sys->b_active == !b_active )
  129.         return VLC_SUCCESS;
  130.     if( b_active )
  131.         return Start( s, psz_extension );
  132.     else
  133.         return Stop( s );
  134. }
  135. /****************************************************************************
  136.  * Helpers
  137.  ****************************************************************************/
  138. static int Start( stream_t *s, const char *psz_extension )
  139. {
  140.     stream_sys_t *p_sys = s->p_sys;
  141.     char *psz_file;
  142.     FILE *f;
  143.     /* */
  144.     if( !psz_extension )
  145.         psz_extension = "dat";
  146.     /* Retreive path */
  147.     char *psz_path = var_CreateGetString( s, "input-record-path" );
  148.     if( !psz_path || *psz_path == '' )
  149.     {
  150.         free( psz_path );
  151.         psz_path = strdup( config_GetHomeDir() );
  152.     }
  153.     if( !psz_path )
  154.         return VLC_ENOMEM;
  155.     /* Create file name
  156.      * TODO allow prefix configuration */
  157.     psz_file = input_CreateFilename( VLC_OBJECT(s), psz_path, INPUT_RECORD_PREFIX, psz_extension );
  158.     free( psz_path );
  159.     if( !psz_file )
  160.         return VLC_ENOMEM;
  161.     f = utf8_fopen( psz_file, "wb" );
  162.     if( !f )
  163.     {
  164.         free( psz_file );
  165.         return VLC_EGENERIC;
  166.     }
  167.     msg_Dbg( s, "Recording into %s", psz_file );
  168.     free( psz_file );
  169.     /* */
  170.     p_sys->f = f;
  171.     p_sys->b_active = true;
  172.     p_sys->b_error = false;
  173.     return VLC_SUCCESS;
  174. }
  175. static int Stop( stream_t *s )
  176. {
  177.     stream_sys_t *p_sys = s->p_sys;
  178.     assert( p_sys->b_active );
  179.     msg_Dbg( s, "Recording completed" );
  180.     fclose( p_sys->f );
  181.     p_sys->b_active = false;
  182.     return VLC_SUCCESS;
  183. }
  184. static void Write( stream_t *s, const uint8_t *p_buffer, size_t i_buffer )
  185. {
  186.     stream_sys_t *p_sys = s->p_sys;
  187.     assert( p_sys->b_active );
  188.     if( i_buffer > 0 )
  189.     {
  190.         const bool b_previous_error = p_sys->b_error;
  191.         const size_t i_written = fwrite( p_buffer, 1, i_buffer, p_sys->f );
  192.         p_sys->b_error = i_written != i_buffer;
  193.         /* TODO maybe a intf_UserError or something like that ? */
  194.         if( p_sys->b_error && !b_previous_error )
  195.             msg_Err( s, "Failed to record data (begin)" );
  196.         else if( !p_sys->b_error && b_previous_error )
  197.             msg_Err( s, "Failed to record data (end)" );
  198.     }
  199. }