]> git.sesse.net Git - vlc/blob - modules/demux/rawdv.c
decoder: do not wait for buffering when there is no data
[vlc] / modules / demux / rawdv.c
1 /*****************************************************************************
2  * rawdv.c : raw DV input module for vlc
3  *****************************************************************************
4  * Copyright (C) 2001-2007 VLC authors and VideoLAN
5  * $Id$
6  *
7  * Authors: Gildas Bazin <gbazin@netcourrier.com>
8  *          Paul Corke <paul dot corke at datatote dot co dot uk>
9  *
10  * This program is free software; you can redistribute it and/or modify it
11  * under the terms of the GNU Lesser General Public License as published by
12  * the Free Software Foundation; either version 2.1 of the License, or
13  * (at your option) any later version.
14  *
15  * This program is distributed in the hope that it will be useful,
16  * but WITHOUT ANY WARRANTY; without even the implied warranty of
17  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18  * GNU Lesser General Public License for more details.
19  *
20  * You should have received a copy of the GNU Lesser General Public License
21  * along with this program; if not, write to the Free Software Foundation,
22  * Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
23  *****************************************************************************/
24
25 /*****************************************************************************
26  * Preamble
27  *****************************************************************************/
28
29 #ifdef HAVE_CONFIG_H
30 # include "config.h"
31 #endif
32
33 #include <vlc_common.h>
34 #include <vlc_plugin.h>
35 #include <vlc_demux.h>
36
37 #include "rawdv.h"
38
39 /*****************************************************************************
40  * Module descriptor
41  *****************************************************************************/
42 #define HURRYUP_TEXT N_( "Hurry up" )
43 #define HURRYUP_LONGTEXT N_( "The demuxer will advance timestamps if the " \
44                 "input can't keep up with the rate." )
45
46 static int  Open ( vlc_object_t * );
47 static void Close( vlc_object_t * );
48
49 vlc_module_begin ()
50     set_shortname( "DV" )
51     set_description( N_("DV (Digital Video) demuxer") )
52     set_capability( "demux", 3 )
53     set_category( CAT_INPUT )
54     set_subcategory( SUBCAT_INPUT_DEMUX )
55     add_bool( "rawdv-hurry-up", false, HURRYUP_TEXT, HURRYUP_LONGTEXT, false )
56     set_callbacks( Open, Close )
57     add_shortcut( "rawdv" )
58 vlc_module_end ()
59
60
61 /*****************************************************************************
62  A little bit of background information (copied over from libdv glossary).
63
64  - DIF block: A block of 80 bytes. This is the basic data framing unit of the
65        DVC tape format, analogous to sectors of hard disc.
66
67  - Video Section: Each DIF sequence contains a video section, consisting of
68        135 DIF blocks, which are further subdivided into Video Segments.
69
70  - Video Segment: A video segment consists of 5 DIF blocks, each corresponding
71        to a single compressed macroblock.
72
73 *****************************************************************************/
74
75 /*****************************************************************************
76  * Definitions of structures used by this plugin
77  *****************************************************************************/
78 typedef struct {
79     int8_t sct;      /* Section type (header,subcode,aux,audio,video) */
80     int8_t dsn;      /* DIF sequence number (0-12) */
81     int    fsc;      /* First (0)/Second channel (1) */
82     int8_t dbn;      /* DIF block number (0-134) */
83 } dv_id_t;
84
85 typedef struct {
86     int    dsf;      /* DIF sequence flag: 525/60 (0) or 625,50 (1) */
87     int8_t apt;
88     int    tf1;
89     int8_t ap1;
90     int    tf2;
91     int8_t ap2;
92     int    tf3;
93     int8_t ap3;
94 } dv_header_t;
95
96 struct demux_sys_t
97 {
98     int    frame_size;
99
100     es_out_id_t *p_es_video;
101     es_format_t  fmt_video;
102
103     es_out_id_t *p_es_audio;
104     es_format_t  fmt_audio;
105
106     int    i_dsf;
107     double f_rate;
108     int    i_bitrate;
109
110     /* program clock reference (in units of 90kHz) */
111     mtime_t i_pcr;
112     bool b_hurry_up;
113 };
114
115 /*****************************************************************************
116  * Local prototypes
117  *****************************************************************************/
118 static int Demux( demux_t * );
119 static int Control( demux_t *, int i_query, va_list args );
120
121 /*****************************************************************************
122  * Open: initializes raw DV demux structures
123  *****************************************************************************/
124 static int Open( vlc_object_t * p_this )
125 {
126     demux_t     *p_demux = (demux_t*)p_this;
127     demux_sys_t *p_sys;
128
129     const uint8_t *p_peek, *p_peek_backup;
130
131     uint32_t    i_dword;
132     dv_header_t dv_header;
133     dv_id_t     dv_id;
134
135     /* It isn't easy to recognize a raw DV stream. The chances that we'll
136      * mistake a stream from another type for a raw DV stream are too high, so
137      * we'll rely on the file extension to trigger this demux. Alternatively,
138      * it is possible to force this demux. */
139
140     /* Check for DV file extension */
141     if( !demux_IsPathExtension( p_demux, ".dv" ) && !p_demux->b_force )
142         return VLC_EGENERIC;
143
144     if( stream_Peek( p_demux->s, &p_peek, DV_PAL_FRAME_SIZE ) <
145         DV_NTSC_FRAME_SIZE )
146     {
147         /* Stream too short ... */
148         msg_Err( p_demux, "cannot peek()" );
149         return VLC_EGENERIC;
150     }
151     p_peek_backup = p_peek;
152
153     /* fill in the dv_id_t structure */
154     i_dword = GetDWBE( p_peek ); p_peek += 4;
155     dv_id.sct = i_dword >> (32 - 3);
156     i_dword <<= 8;
157     dv_id.dsn = i_dword >> (32 - 4);
158     i_dword <<= 4;
159     dv_id.fsc = i_dword >> (32 - 1);
160     i_dword <<= 4;
161     dv_id.dbn = i_dword >> (32 - 8);
162     i_dword <<= 8;
163
164     if( dv_id.sct != 0 )
165     {
166         msg_Warn( p_demux, "not a raw DV stream header" );
167         return VLC_EGENERIC;
168     }
169
170     /* fill in the dv_header_t structure */
171     dv_header.dsf = i_dword >> (32 - 1);
172     i_dword <<= 1;
173     if( i_dword >> (32 - 1) ) /* incorrect bit */
174     {
175         msg_Warn( p_demux, "incorrect bit" );
176         return VLC_EGENERIC;
177     }
178
179     i_dword = GetDWBE( p_peek ); p_peek += 4;
180     i_dword <<= 5;
181     dv_header.apt = i_dword >> (32 - 3);
182     i_dword <<= 3;
183     dv_header.tf1 = i_dword >> (32 - 1);
184     i_dword <<= 5;
185     dv_header.ap1 = i_dword >> (32 - 3);
186     i_dword <<= 3;
187     dv_header.tf2 = i_dword >> (32 - 1);
188     i_dword <<= 5;
189     dv_header.ap2 = i_dword >> (32 - 3);
190     i_dword <<= 3;
191     dv_header.tf3 = i_dword >> (32 - 1);
192     i_dword <<= 5;
193     dv_header.ap3 = i_dword >> (32 - 3);
194
195     p_peek += 72;                                  /* skip rest of DIF block */
196
197     p_demux->p_sys      = p_sys = malloc( sizeof( demux_sys_t ) );
198     if( !p_sys )
199         return VLC_ENOMEM;
200
201     p_sys->b_hurry_up = var_CreateGetBool( p_demux, "rawdv-hurry-up" );
202     msg_Dbg( p_demux, "Realtime DV Source: %s", (p_sys->b_hurry_up)?"Yes":"No" );
203
204     p_sys->i_dsf = dv_header.dsf;
205     p_sys->frame_size = dv_header.dsf ? DV_PAL_FRAME_SIZE
206                                       : DV_NTSC_FRAME_SIZE;
207     p_sys->f_rate = dv_header.dsf ? 25 : 29.97;
208
209     p_sys->i_pcr = 0;
210     p_sys->p_es_video = NULL;
211     p_sys->p_es_audio = NULL;
212
213     p_sys->i_bitrate = 0;
214
215     es_format_Init( &p_sys->fmt_video, VIDEO_ES, VLC_CODEC_DV );
216     p_sys->fmt_video.video.i_width = 720;
217     p_sys->fmt_video.video.i_height= dv_header.dsf ? 576 : 480;;
218
219     p_sys->p_es_video = es_out_Add( p_demux->out, &p_sys->fmt_video );
220
221     /* Audio stuff */
222     p_peek = p_peek_backup + 80*6+80*16*3 + 3; /* beginning of AAUX pack */
223     if( *p_peek == 0x50 )
224     {
225         dv_get_audio_format( &p_sys->fmt_audio, &p_peek[1] );
226         p_sys->p_es_audio = es_out_Add( p_demux->out, &p_sys->fmt_audio );
227     }
228
229     p_demux->pf_demux   = Demux;
230     p_demux->pf_control = Control;
231     return VLC_SUCCESS;
232 }
233
234 /*****************************************************************************
235  * Close: frees unused data
236  *****************************************************************************/
237 static void Close( vlc_object_t *p_this )
238 {
239     demux_t     *p_demux = (demux_t*)p_this;
240     demux_sys_t *p_sys  = p_demux->p_sys;
241
242     var_Destroy( p_demux, "rawdv-hurry-up");
243     free( p_sys );
244 }
245
246 /*****************************************************************************
247  * Demux: reads and demuxes data packets
248  *****************************************************************************
249  * Returns -1 in case of error, 0 in case of EOF, 1 otherwise
250  *****************************************************************************/
251 static int Demux( demux_t *p_demux )
252 {
253     demux_sys_t *p_sys  = p_demux->p_sys;
254     block_t     *p_block;
255     bool  b_audio = false;
256
257     if( p_sys->b_hurry_up )
258     {
259          /* 3 frames */
260         p_sys->i_pcr = mdate() + (p_sys->i_dsf ? 120000 : 90000);
261     }
262
263     /* Call the pace control */
264     es_out_Control( p_demux->out, ES_OUT_SET_PCR, VLC_TS_0 + p_sys->i_pcr );
265     p_block = stream_Block( p_demux->s, p_sys->frame_size );
266     if( p_block == NULL )
267     {
268         /* EOF */
269         return 0;
270     }
271
272     if( p_sys->p_es_audio )
273     {
274         es_out_Control( p_demux->out, ES_OUT_GET_ES_STATE,
275                         p_sys->p_es_audio, &b_audio );
276     }
277
278     p_block->i_dts =
279     p_block->i_pts = VLC_TS_0 + p_sys->i_pcr;
280
281     if( b_audio )
282     {
283         block_t *p_audio_block = dv_extract_audio( p_block );
284         if( p_audio_block )
285             es_out_Send( p_demux->out, p_sys->p_es_audio, p_audio_block );
286     }
287
288     es_out_Send( p_demux->out, p_sys->p_es_video, p_block );
289
290     if( !p_sys->b_hurry_up )
291     {
292         p_sys->i_pcr += ( INT64_C(1000000) / p_sys->f_rate );
293     }
294
295     return 1;
296 }
297
298 /*****************************************************************************
299  * Control:
300  *****************************************************************************/
301 static int Control( demux_t *p_demux, int i_query, va_list args )
302 {
303     demux_sys_t *p_sys  = p_demux->p_sys;
304
305     /* XXX: DEMUX_SET_TIME is precise here */
306     return demux_vaControlHelper( p_demux->s,
307                                    0, -1,
308                                    p_sys->frame_size * p_sys->f_rate * 8,
309                                    p_sys->frame_size, i_query, args );
310 }
311