1 /* access.c: DVD access plugin.
2 *****************************************************************************
3 * This plugins should handle all the known specificities of the DVD format,
4 * especially the 2048 bytes logical block size.
6 * -libdvdcss for access and unscrambling
7 * -ifo.* for ifo parsing and analyse
9 *****************************************************************************
10 * Copyright (C) 1998-2001 VideoLAN
11 * $Id: access.c,v 1.12 2003/03/24 17:15:29 gbazin Exp $
13 * Author: Stéphane Borel <stef@via.ecp.fr>
15 * This program is free software; you can redistribute it and/or modify
16 * it under the terms of the GNU General Public License as published by
17 * the Free Software Foundation; either version 2 of the License, or
18 * (at your option) any later version.
20 * This program is distributed in the hope that it will be useful,
21 * but WITHOUT ANY WARRANTY; without even the implied warranty of
22 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
23 * GNU General Public License for more details.
25 * You should have received a copy of the GNU General Public License
26 * along with this program; if not, write to the Free Software
27 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111, USA.
28 *****************************************************************************/
30 /*****************************************************************************
32 *****************************************************************************/
38 #include <vlc/input.h>
40 #include "../../demux/mpeg/system.h"
47 #include <sys/types.h>
52 #ifdef STRNCASECMP_IN_STRINGS_H
59 # include <dvdcss/dvdcss.h>
69 /*****************************************************************************
71 *****************************************************************************/
73 /* called from outside */
74 static int DVDSetArea ( input_thread_t *, input_area_t * );
75 static int DVDSetProgram ( input_thread_t *, pgrm_descriptor_t * );
76 static ssize_t DVDRead ( input_thread_t *, byte_t *, size_t );
77 static void DVDSeek ( input_thread_t *, off_t );
79 static char * DVDParse( input_thread_t * );
82 * Data access functions
85 #define DVDTell LB2OFF( p_dvd->i_vts_start + p_dvd->i_vts_lb ) \
86 - p_input->stream.p_selected_area->i_start
88 /*****************************************************************************
90 *****************************************************************************/
91 int E_(DVDOpen) ( vlc_object_t *p_this )
93 input_thread_t * p_input = (input_thread_t *)p_this;
95 thread_dvd_data_t * p_dvd;
96 input_area_t * p_area;
98 char * psz_dvdcss_env;
100 p_dvd = malloc( sizeof(thread_dvd_data_t) );
103 msg_Err( p_input, "out of memory" );
106 p_input->p_access_data = (void *)p_dvd;
108 p_input->pf_read = DVDRead;
109 p_input->pf_seek = DVDSeek;
110 p_input->pf_set_area = DVDSetArea;
111 p_input->pf_set_program = DVDSetProgram;
113 /* Parse command line */
114 if( !( psz_device = DVDParse( p_input ) ) )
125 /* override environment variable DVDCSS_METHOD with config option
126 * (FIXME: this creates a small memory leak) */
127 psz_dvdcss_env = config_GetPsz( p_input, "dvdcss-method" );
128 if( psz_dvdcss_env && *psz_dvdcss_env )
132 psz_env = malloc( strlen("DVDCSS_METHOD=") +
133 strlen( psz_dvdcss_env ) + 1 );
141 sprintf( psz_env, "%s%s", "DVDCSS_METHOD=", psz_dvdcss_env );
145 if( psz_dvdcss_env ) free( psz_dvdcss_env );
150 p_dvd->dvdhandle = dvdcss_open( psz_device );
152 /* free allocated string */
155 if( p_dvd->dvdhandle == NULL )
157 msg_Err( p_input, "dvdcss cannot open device" );
162 if( dvdcss_seek( p_dvd->dvdhandle, 0, DVDCSS_NOFLAGS ) < 0 )
164 msg_Err( p_input, "%s", dvdcss_error( p_dvd->dvdhandle ) );
165 dvdcss_close( p_dvd->dvdhandle );
170 /* Ifo allocation & initialisation */
171 if( IfoCreate( p_dvd ) < 0 )
173 msg_Err( p_input, "allcation error in ifo" );
174 dvdcss_close( p_dvd->dvdhandle );
179 if( IfoInit( p_dvd->p_ifo ) < 0 )
181 msg_Err( p_input, "fatal failure in ifo" );
182 IfoDestroy( p_dvd->p_ifo );
183 dvdcss_close( p_dvd->dvdhandle );
188 /* Set stream and area data */
189 vlc_mutex_lock( &p_input->stream.stream_lock );
191 p_input->stream.i_method = INPUT_METHOD_DVD;
192 p_input->stream.b_pace_control = 1;
193 p_input->stream.b_seekable = 1;
194 p_input->stream.p_selected_area->i_size = 0;
195 p_input->stream.p_selected_area->i_tell = 0;
197 /* Initialize ES structures */
198 input_InitStream( p_input, sizeof( stream_ps_data_t ) );
200 #define title_inf p_dvd->p_ifo->vmg.title_inf
201 msg_Dbg( p_input, "number of titles: %d", title_inf.i_title_nb );
203 #define area p_input->stream.pp_areas
204 /* We start from 1 here since the default area 0
205 * is reserved for video_ts.vob */
206 for( i = 1 ; i <= title_inf.i_title_nb ; i++ )
208 /* Titles are Program Chains */
209 input_AddArea( p_input, i, title_inf.p_attr[i-1].i_chapter_nb );
211 /* Absolute start offset and size
212 * We can only set that with vts ifo, so we do it during the
213 * first call to DVDSetArea */
214 area[i]->i_start = 0;
217 /* Default Chapter */
220 /* Offset to vts_i_0.ifo */
221 area[i]->i_plugin_data = p_dvd->p_ifo->i_start +
222 title_inf.p_attr[i-1].i_start_sector;
226 p_dvd->i_title = p_dvd->i_title <= title_inf.i_title_nb ?
230 p_area = p_input->stream.pp_areas[p_dvd->i_title];
232 p_area->i_part = p_dvd->i_chapter <= p_area->i_part_nb ?
233 p_dvd->i_chapter : 1;
234 p_dvd->i_chapter = 1;
236 p_dvd->b_new_chapter = 0;
237 p_dvd->i_audio_nb = 0;
240 /* set title, chapter, audio and subpic */
241 if( DVDSetArea( p_input, p_area ) < 0 )
243 vlc_mutex_unlock( &p_input->stream.stream_lock );
244 IfoDestroy( p_dvd->p_ifo );
245 dvdcss_close( p_dvd->dvdhandle );
250 vlc_mutex_unlock( &p_input->stream.stream_lock );
252 if( !p_input->psz_demux || !*p_input->psz_demux )
254 p_input->psz_demux = "dvdold";
260 /*****************************************************************************
261 * DVDClose: close dvd
262 *****************************************************************************/
263 void E_(DVDClose) ( vlc_object_t *p_this )
265 input_thread_t * p_input = (input_thread_t *)p_this;
266 thread_dvd_data_t *p_dvd = (thread_dvd_data_t*)p_input->p_access_data;
268 /* This is a very nasty side-effect in the DVD plug-in : language
269 * selection here influences language selection of other streams. So
270 * unset those variables (may not be what the user wants).
271 * FIXME FIXME FIXME FIXME FIXME FIXME FIXME --Meuuh */
272 config_PutInt( p_input, "audio-channel", -1 );
273 config_PutInt( p_input, "spu-channel", -1 );
275 IfoDestroy( p_dvd->p_ifo );
276 dvdcss_close( p_dvd->dvdhandle );
280 /*****************************************************************************
281 * DVDSetProgram: used to change angle
282 *****************************************************************************/
283 static int DVDSetProgram( input_thread_t * p_input,
284 pgrm_descriptor_t * p_program )
286 if( p_input->stream.p_selected_program != p_program )
288 thread_dvd_data_t * p_dvd;
292 p_dvd = (thread_dvd_data_t*)(p_input->p_access_data);
293 i_angle = p_program->i_number;
295 /* DVD is actually mono-program: we only need the current angle
296 * number, so copy the data between programs */
298 p_input->stream.p_selected_program,
299 sizeof(pgrm_descriptor_t) );
300 p_program->i_number = i_angle;
301 p_input->stream.p_selected_program = p_program;
304 p_dvd->p_ifo->vts.title_unit.p_title[p_dvd->i_title_id-1].title
305 if( title.p_cell_play[p_dvd->i_prg_cell].i_category & 0xf000 )
307 if( ( p_program->i_number - p_dvd->i_angle ) < 0 )
309 /* we have to go backwards */
310 p_dvd->i_map_cell = 0;
312 p_dvd->i_prg_cell += ( p_program->i_number - p_dvd->i_angle );
313 p_dvd->i_map_cell = CellPrg2Map( p_dvd );
314 p_dvd->i_map_cell += p_dvd->i_angle_cell;
315 p_dvd->i_vts_lb = CellFirstSector( p_dvd );
316 p_dvd->i_last_lb = CellLastSector( p_dvd );
317 p_dvd->i_angle = p_program->i_number;
321 p_dvd->i_angle = p_program->i_number;
324 msg_Dbg( p_input, "angle %d selected", p_dvd->i_angle );
326 /* Update the navigation variables without triggering a callback */
327 val.i_int = p_program->i_number;
328 var_Change( p_input, "program", VLC_VAR_SETVALUE, &val );
334 /*****************************************************************************
335 * DVDSetArea: initialize input data for title x, chapter y.
336 * It should be called for each user navigation request.
337 *****************************************************************************
338 * Take care that i_title starts from 0 (vmg) and i_chapter start from 1.
339 * Note that you have to take the lock before entering here.
340 *****************************************************************************/
341 #define vmg p_dvd->p_ifo->vmg
342 #define vts p_dvd->p_ifo->vts
344 static void DVDFlushStream( input_thread_t * p_input )
346 if( p_input->stream.pp_programs != NULL )
348 /* We don't use input_EndStream here since
349 * we keep area structures */
350 while( p_input->stream.i_es_number )
352 input_DelES( p_input, p_input->stream.pp_es[0] );
355 while( p_input->stream.i_pgrm_number )
357 input_DelProgram( p_input, p_input->stream.pp_programs[0] );
360 if( p_input->stream.pp_selected_es )
362 free( p_input->stream.pp_selected_es );
363 p_input->stream.pp_selected_es = NULL;
365 p_input->stream.i_selected_es_number = 0;
371 static int DVDReadAngle( input_thread_t * p_input )
373 thread_dvd_data_t * p_dvd;
377 p_dvd = (thread_dvd_data_t*)(p_input->p_access_data);
378 i_angle_nb = vmg.title_inf.p_attr[p_dvd->i_title-1].i_angle_nb;
380 input_AddProgram( p_input, 1, sizeof( stream_ps_data_t ) );
381 p_input->stream.p_selected_program = p_input->stream.pp_programs[0];
383 for( i = 1 ; i < i_angle_nb ; i++ )
385 input_AddProgram( p_input, i+1, 0 );
391 static int DVDSetArea( input_thread_t * p_input, input_area_t * p_area )
393 thread_dvd_data_t * p_dvd;
396 p_dvd = (thread_dvd_data_t*)(p_input->p_access_data);
398 /* we can't use the interface slider until initilization is complete */
399 p_input->stream.b_seekable = 0;
401 if( p_area != p_input->stream.p_selected_area )
408 /* Reset the Chapter position of the old title */
409 p_input->stream.p_selected_area->i_part = 1;
410 p_input->stream.p_selected_area = p_area;
413 * We have to load all title information
416 /* title number as it appears in the interface list */
417 p_dvd->i_title = p_area->i_id;
418 p_dvd->i_chapter_nb = p_area->i_part_nb;
420 if( IfoTitleSet( p_dvd->p_ifo, p_dvd->i_title ) < 0 )
422 msg_Err( p_input, "fatal error in vts ifo" );
427 /* title position inside the selected vts */
428 i_vts_title = vmg.title_inf.p_attr[p_dvd->i_title-1].i_title_num;
430 vts.title_inf.p_title_start[i_vts_title-1].i_title_id;
432 msg_Dbg( p_input, "title %d vts_title %d pgc %d",
433 p_dvd->i_title, i_vts_title, p_dvd->i_title_id );
435 /* title set offset XXX: convert to block values */
437 vts.i_pos + vts.manager_inf.i_title_vob_start_sector;
440 p_dvd->i_prg_cell = -1 +
441 vts.title_unit.p_title[p_dvd->i_title_id-1].title.i_cell_nb;
442 p_dvd->i_map_cell = 0;
443 p_dvd->i_map_cell = CellPrg2Map( p_dvd );
444 i_last = CellLastSector( p_dvd );
447 p_dvd->i_prg_cell = 0;
448 p_dvd->i_map_cell = 0;
449 p_dvd->i_angle_cell = 0;
450 p_dvd->i_map_cell = CellPrg2Map ( p_dvd );
451 p_dvd->i_vts_lb = CellFirstSector( p_dvd );
452 p_dvd->i_last_lb = CellLastSector ( p_dvd );
454 /* Force libdvdcss to check its title key.
455 * It is only useful for title cracking method. Methods using the
456 * decrypted disc key are fast enough to check the key at each seek */
457 i_first = dvdcss_seek( p_dvd->dvdhandle,
458 p_dvd->i_vts_start + p_dvd->i_vts_lb,
462 msg_Err( p_input, "%s", dvdcss_error( p_dvd->dvdhandle ) );
466 /* Area definition */
467 p_input->stream.p_selected_area->i_start = LB2OFF( i_first );
468 p_input->stream.p_selected_area->i_size =
469 LB2OFF( i_last + 1 - p_dvd->i_vts_lb );
471 /* Destroy obsolete ES by reinitializing programs */
472 DVDFlushStream( p_input );
474 /* Angle management: angles are handled through programs */
475 p_dvd->i_angle_nb = DVDReadAngle( p_input );
476 if( ( p_dvd->i_angle <= 0 ) || p_dvd->i_angle > p_dvd->i_angle_nb )
481 DVDSetProgram( p_input,
482 p_input->stream.pp_programs[p_dvd->i_angle-1] );
484 msg_Dbg( p_input, "title first %i, last %i, size %i",
485 i_first, i_last, i_last + 1 - p_dvd->i_vts_lb );
486 IfoPrintTitle( p_dvd );
488 /* No PSM to read in DVD mode, we already have all information */
489 p_input->stream.p_selected_program->b_is_ok = 1;
491 /* Find all ES in title with ifo data */
492 DVDReadVideo( p_input );
493 DVDReadAudio( p_input );
494 DVDReadSPU ( p_input );
496 if( p_input->p_demux )
498 DVDLaunchDecoders( p_input );
501 /* Update the navigation variables without triggering a callback */
502 val.i_int = p_area->i_id;
503 var_Change( p_input, "title", VLC_VAR_SETVALUE, &val );
504 var_Change( p_input, "chapter", VLC_VAR_CLEARCHOICES, NULL );
505 for( i = 1; i <= p_area->i_part_nb; i++ )
508 var_Change( p_input, "chapter", VLC_VAR_ADDCHOICE, &val );
514 p_area = p_input->stream.p_selected_area;
517 /* Chapter selection */
518 p_dvd->i_chapter = DVDSetChapter( p_dvd, p_area->i_part );
520 p_input->stream.p_selected_area->i_tell = DVDTell;
522 /* warn interface that something has changed */
523 p_input->stream.b_seekable = 1;
524 p_input->stream.b_changed = 1;
526 /* Update the navigation variables without triggering a callback */
527 val.i_int = p_area->i_part;
528 var_Change( p_input, "chapter", VLC_VAR_SETVALUE, &val );
536 p_dvd->p_ifo->vts.title_unit.p_title[p_dvd->i_title_id-1].title
538 /*****************************************************************************
539 * DVDRead: reads data packets.
540 *****************************************************************************
541 * Returns -1 in case of error, 0 in case of EOF, otherwise the number of
543 *****************************************************************************/
544 static ssize_t DVDRead( input_thread_t * p_input,
545 byte_t * p_buffer, size_t i_count )
547 thread_dvd_data_t * p_dvd;
550 int i_block_once = 0;
552 p_dvd = (thread_dvd_data_t *)(p_input->p_access_data);
555 i_blocks = OFF2LB(i_count);
559 i_block_once = LbMaxOnce( p_dvd );
560 if( i_block_once > i_blocks )
562 i_block_once = i_blocks;
564 else if( i_block_once <= 0 )
570 if( i_block_once != dvdcss_read( p_dvd->dvdhandle, p_buffer,
571 i_block_once, DVDCSS_READ_DECRYPT ) )
576 i_blocks -= i_block_once;
577 i_read += i_block_once;
578 p_buffer += LB2OFF( i_block_once );
580 /* Update global position */
581 p_dvd->i_vts_lb += i_block_once;
584 vlc_mutex_lock( &p_input->stream.stream_lock );
586 if( p_dvd->b_new_chapter )
588 p_input->stream.p_selected_area->i_part = p_dvd->i_chapter;
589 p_dvd->b_new_chapter = 0;
592 if( ( p_input->stream.p_selected_area->i_tell + LB2OFF( i_read )
593 >= p_input->stream.p_selected_area->i_size )
594 || ( i_block_once <= 0 ) )
596 if( ( p_dvd->i_title + 1 ) >= p_input->stream.i_area_nb )
599 vlc_mutex_unlock( &p_input->stream.stream_lock );
604 msg_Dbg( p_input, "new title" );
606 DVDSetArea( p_input, p_input->stream.pp_areas[p_dvd->i_title] );
609 vlc_mutex_unlock( &p_input->stream.stream_lock );
611 return LB2OFF( i_read );
614 /*****************************************************************************
615 * DVDSeek : Goes to a given position on the stream.
616 *****************************************************************************
617 * This one is used by the input and translate chronological position from
618 * input to logical position on the device.
619 * The lock should be taken before calling this function.
620 *****************************************************************************/
621 static void DVDSeek( input_thread_t * p_input, off_t i_off )
623 thread_dvd_data_t * p_dvd;
625 p_dvd = ( thread_dvd_data_t * )(p_input->p_access_data);
627 vlc_mutex_lock( &p_input->stream.stream_lock );
628 p_dvd->i_vts_lb = OFF2LB(i_off + p_input->stream.p_selected_area->i_start)
629 - p_dvd->i_vts_start;
630 vlc_mutex_unlock( &p_input->stream.stream_lock );
632 p_dvd->i_prg_cell = Lb2CellPrg( p_dvd );
633 p_dvd->i_map_cell = Lb2CellMap( p_dvd );
635 if( CellIsInterleaved( p_dvd ) )
637 /* if we're inside a multi-angle zone, we have to choose i_sector
638 * in the current angle ; we can't do it all the time since cells
639 * can be very wide out of such zones */
640 p_dvd->i_vts_lb = CellFirstSector( p_dvd );
643 p_dvd->i_last_lb = CellLastSector( p_dvd );
644 p_dvd->i_chapter = CellPrg2Chapter( p_dvd );
646 if( dvdcss_seek( p_dvd->dvdhandle, p_dvd->i_vts_start + p_dvd->i_vts_lb,
647 DVDCSS_SEEK_MPEG ) < 0 )
649 msg_Err( p_input, "%s", dvdcss_error( p_dvd->dvdhandle ) );
650 p_input->b_error = 1;
654 vlc_mutex_lock( &p_input->stream.stream_lock );
655 p_input->stream.p_selected_area->i_part = p_dvd->i_chapter;
656 p_input->stream.p_selected_area->i_tell = DVDTell;
657 vlc_mutex_unlock( &p_input->stream.stream_lock );
659 msg_Dbg( p_input, "program cell: %d cell: %d chapter: %d tell "I64Fd,
660 p_dvd->i_prg_cell, p_dvd->i_map_cell, p_dvd->i_chapter, DVDTell );
665 /*****************************************************************************
666 * DVDParse: parse command line
667 *****************************************************************************/
668 static char * DVDParse( input_thread_t * p_input )
670 thread_dvd_data_t * p_dvd;
671 struct stat stat_info;
676 vlc_bool_t b_options = 0;
682 p_dvd = (thread_dvd_data_t*)(p_input->p_access_data);
685 /* On Win32 we want the DVD access plugin to be explicitly requested,
686 * we end up with lots of problems otherwise */
687 if( !p_input->psz_access || !*p_input->psz_access ) return NULL;
690 psz_parser = psz_device = strdup( p_input->psz_name );
696 /* Parse input string :
697 * [device][@rawdevice][@[title][,[chapter][,angle]]] */
698 while( *psz_parser && *psz_parser != '@' )
703 if( *psz_parser == '@' )
705 /* Maybe found raw device or option list */
707 psz_raw = ++psz_parser;
714 if( *psz_parser && !strtol( psz_parser, NULL, 10 ) )
716 /* what we've found is either a raw device or a partial option
717 * list e.g. @,29 or both a device and a list ; search end of string */
718 while( *psz_parser && *psz_parser != '@' )
723 if( *psz_parser == '@' )
725 /* found end of raw device, and beginning of options */
732 psz_parser = psz_raw + 1;
733 for( i=0 ; i<3 ; i++ )
737 /* we have only a raw device */
740 if( strtol( psz_parser, NULL, 10 ) )
742 /* we have only a partial list of options, no device */
743 psz_parser = psz_raw;
754 /* found beginning of options ; no raw device specified */
762 i_title = (int)strtol( psz_parser, &psz_next, 10 );
765 psz_parser = psz_next + 1;
766 i_chapter = (int)strtol( psz_parser, &psz_next, 10 );
769 i_angle = (int)strtol( psz_next + 1, NULL, 10 );
773 p_dvd->i_title = i_title ? i_title : 1;
774 p_dvd->i_chapter = i_chapter ? i_chapter : 1;
775 p_dvd->i_angle = i_angle ? i_angle : 1;
782 /* check the raw device */
783 if( stat( psz_raw, &stat_info ) == -1 )
785 msg_Warn( p_input, "cannot stat() raw device `%s' (%s)",
786 psz_raw, strerror(errno));
788 *(psz_raw - 1) = '@';
796 if( !S_ISCHR(stat_info.st_mode) )
798 msg_Warn( p_input, "raw device %s is"
799 " not a valid char device", psz_raw );
801 *(psz_raw - 1) = '@';
807 psz_env = malloc( strlen("DVDCSS_RAW_DEVICE=")
808 + strlen( psz_raw ) + 1 );
809 sprintf( psz_env, "DVDCSS_RAW_DEVICE=%s", psz_raw );
824 if( !p_input->psz_access )
826 /* no device and no access specified: we probably don't want DVD */
829 psz_device = config_GetPsz( p_input, "dvd" );
833 /* check block device */
834 if( stat( psz_device, &stat_info ) == -1 )
836 msg_Warn( p_input, "cannot stat() device `%s' (%s)",
837 psz_device, strerror(errno));
842 if( !S_ISBLK(stat_info.st_mode) && !S_ISCHR(stat_info.st_mode) )
845 "dvd module discarded (not a valid block device)" );
851 msg_Dbg( p_input, "dvd=%s raw=%s title=%d chapter=%d angle=%d",
852 psz_device, psz_raw, p_dvd->i_title,
853 p_dvd->i_chapter, p_dvd->i_angle );