]> git.sesse.net Git - vlc/blob - modules/demux/demuxdump.c
* include/configuration.h: added a new flag to the configuration stucture to
[vlc] / modules / demux / demuxdump.c
1 /*****************************************************************************
2  * demuxdump.c : Pseudo demux module for vlc (dump raw stream)
3  *****************************************************************************
4  * Copyright (C) 2001 VideoLAN
5  * $Id: demuxdump.c,v 1.5 2003/02/20 01:52:46 sigmunau Exp $
6  *
7  * Authors: Laurent Aimar <fenrir@via.ecp.fr>
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., 59 Temple Place - Suite 330, Boston, MA  02111, USA.
22  *****************************************************************************/
23
24 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27 #include <stdlib.h>                                      /* malloc(), free() */
28 #include <string.h>                                              /* strdup() */
29 #include <errno.h>
30
31 #include <vlc/vlc.h>
32 #include <vlc/input.h>
33
34 #include <sys/types.h>
35
36 /*****************************************************************************
37  * Local prototypes
38  *****************************************************************************/
39 static int  Activate ( vlc_object_t * );
40 static int  Demux ( input_thread_t * );
41 static void Desactivate ( vlc_object_t * );
42
43 #define DUMP_BLOCKSIZE  16384
44
45 /*****************************************************************************
46  * Module descriptor
47  *****************************************************************************/
48 vlc_module_begin();
49     set_description( _("Dump Demux input") );
50     set_capability( "demux", 0 );
51     add_category_hint( "File", NULL, VLC_FALSE );
52         add_string( "demuxdump-file", NULL, NULL, 
53                     "dump file name", 
54                     "file name for dumping raw stream read by demux", VLC_FALSE );
55     set_callbacks( Activate, Desactivate );
56     add_shortcut( "dump" );
57 vlc_module_end();
58
59 struct demux_sys_t
60 {
61     char        *psz_name;
62     FILE        *p_file;
63     uint64_t    i_write;
64
65     void        *p_demux_data_sav;
66 };
67
68 /*
69  * Data reading functions
70  */
71
72 /*****************************************************************************
73  * Activate: initializes dump structures
74  *****************************************************************************/
75 static int Activate( vlc_object_t * p_this )
76 {
77     input_thread_t      *p_input = (input_thread_t *)p_this;
78     demux_sys_t         *p_demux;
79
80     char                *psz_name;
81
82     /* Set the demux function */
83     p_input->pf_demux = Demux;
84
85     /* Initialize access plug-in structures. */
86     if( p_input->i_mtu == 0 )
87     {
88         /* Improve speed. */
89         p_input->i_bufsize = INPUT_DEFAULT_BUFSIZE;
90     }
91     
92     psz_name = config_GetPsz( p_input, "demuxdump-file" );
93     if( !psz_name || !*psz_name )
94     {
95         psz_name = strdup( "stream-demux.dump" );
96     }
97
98     p_demux = malloc( sizeof( demux_sys_t ) );
99     memset( p_demux, 0, sizeof( demux_sys_t ) );
100
101     if( !strcmp( psz_name, "-" ) )
102     {
103         msg_Info( p_input,
104                   "dumping raw stream to standard output" );
105         p_demux->p_file = stdout;
106         p_demux->psz_name = psz_name;
107     }
108     else if( !( p_demux->p_file = fopen( psz_name, "wb" ) ) )
109     {
110         msg_Err( p_input,
111                  "cannot create `%s' for writing", 
112                  psz_name );
113         free( p_demux );
114         return( -1 );
115     }
116     else
117     {
118         msg_Info( p_input,
119                   "dumping raw stream to file `%s'", 
120                   psz_name );
121         p_demux->psz_name = psz_name;
122     }
123
124     p_demux->i_write = 0;
125     p_demux->p_demux_data_sav = p_input->p_demux_data;
126
127     if( p_input->stream.p_selected_program != NULL )
128     {
129         /* workaround for dvd access */
130         msg_Warn( p_input, "demux data already initializated (by access?)" );
131     }
132     else
133     {
134
135         if( input_InitStream( p_input, 0 ) == -1 )
136         {
137             if( p_demux->p_file != stdout )
138                 fclose( p_demux->p_file );
139             free( p_demux );
140             return( -1 );
141         }
142         input_AddProgram( p_input, 0, 0 );
143         p_input->stream.p_selected_program = p_input->stream.pp_programs[0];
144
145         vlc_mutex_lock( &p_input->stream.stream_lock );
146         p_input->stream.p_selected_area->i_tell = 0;
147         vlc_mutex_unlock( &p_input->stream.stream_lock );
148     }
149
150     p_input->p_demux_data = p_demux;
151
152     vlc_mutex_lock( &p_input->stream.stream_lock );
153     p_input->stream.p_selected_program->b_is_ok = 1;
154     vlc_mutex_unlock( &p_input->stream.stream_lock );
155     
156     return( 0 );
157 }
158
159 /*****************************************************************************
160  * Desctivate: initializes dump structures
161  *****************************************************************************/
162 static void Desactivate ( vlc_object_t *p_this )
163 {
164     input_thread_t      *p_input = (input_thread_t *)p_this;
165     demux_sys_t         *p_demux = (demux_sys_t*)p_input->p_demux_data;
166
167     msg_Info( p_input,
168               "closing %s ("I64Fd" Kbytes dumped)",
169               p_demux->psz_name,
170               p_demux->i_write / 1024 );
171
172     if( p_demux->p_file )
173     {
174         if( p_demux->p_file != stdout )
175             fclose( p_demux->p_file );
176         p_demux->p_file = NULL;
177     }
178     if( p_demux->psz_name )
179     {
180         free( p_demux->psz_name );
181     }
182     p_input->p_demux_data = p_demux->p_demux_data_sav;
183     free( p_demux );
184 }
185
186 /*****************************************************************************
187  * Demux: reads and demuxes data packets
188  *****************************************************************************
189  * Returns -1 in case of error, 0 in case of EOF, 1 otherwise
190  *****************************************************************************/
191 static int Demux( input_thread_t * p_input )
192 {
193     demux_sys_t         *p_demux = (demux_sys_t*)p_input->p_demux_data;
194
195     ssize_t         i_read;
196     data_packet_t * p_data;
197     int             i_write;
198
199     p_input->p_demux_data = p_demux->p_demux_data_sav;
200     i_read = input_SplitBuffer( p_input, &p_data, DUMP_BLOCKSIZE );
201     p_input->p_demux_data = p_demux;
202     
203     if ( i_read <= 0 )
204     {
205         return i_read;
206     }
207
208     i_write = fwrite( p_data->p_payload_start,
209                        1,
210                        i_read,
211                        p_demux->p_file );
212     
213     input_DeletePacket( p_input->p_method_data, p_data );
214
215     if( i_write < 0 )
216     {
217         msg_Err( p_input, 
218                  "failed to write %d bytes",
219                  i_write );
220         return( -1 );
221     }
222     else
223     {
224         msg_Dbg( p_input,
225                  "dumped %d bytes",
226                  i_write );
227         p_demux->i_write += i_write;
228     }
229
230
231     if( (p_input->stream.p_selected_program->i_synchro_state == SYNCHRO_REINIT)
232          | (input_ClockManageControl( p_input, 
233                       p_input->stream.p_selected_program,
234                          (mtime_t)0 ) == PAUSE_S) )
235     {
236         msg_Warn( p_input, "synchro reinit" );
237         p_input->stream.p_selected_program->i_synchro_state = SYNCHRO_OK;
238     }
239
240     return( 1 );
241 }
242