1 /*****************************************************************************
2 * playlist.c : Playlist management functions
3 *****************************************************************************
4 * Copyright (C) 1999-2001 VideoLAN
5 * $Id: playlist.c,v 1.56 2003/09/24 10:21:32 zorglub Exp $
7 * Authors: Samuel Hocevar <sam@zoy.org>
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.
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.
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 #include <stdlib.h> /* free(), strtol() */
24 #include <stdio.h> /* sprintf() */
25 #include <string.h> /* strerror() */
31 #include "stream_control.h"
32 #include "input_ext-intf.h"
34 #include "vlc_playlist.h"
36 #define PLAYLIST_FILE_HEADER_0_5 "# vlc playlist file version 0.5"
38 /*****************************************************************************
40 *****************************************************************************/
41 static void RunThread ( playlist_t * );
42 static void SkipItem ( playlist_t *, int );
43 static void PlayItem ( playlist_t * );
45 static void Poubellize ( playlist_t *, input_thread_t * );
50 * Create a playlist structure.
51 * \param p_parent the vlc object that is to be the parent of this playlist
52 * \return a pointer to the created playlist, or NULL on error
54 playlist_t * __playlist_Create ( vlc_object_t *p_parent )
56 playlist_t *p_playlist;
59 /* Allocate structure */
60 p_playlist = vlc_object_create( p_parent, VLC_OBJECT_PLAYLIST );
63 msg_Err( p_parent, "out of memory" );
67 var_Create( p_playlist, "intf-change", VLC_VAR_BOOL );
68 val.b_bool = VLC_TRUE;
69 var_Set( p_playlist, "intf-change", val );
71 var_Create( p_playlist, "intf-popupmenu", VLC_VAR_BOOL );
73 var_Create( p_playlist, "intf-show", VLC_VAR_BOOL );
74 val.b_bool = VLC_TRUE;
75 var_Set( p_playlist, "intf-show", val );
77 var_Create( p_playlist, "random", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
78 var_Create( p_playlist, "repeat", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
79 var_Create( p_playlist, "loop", VLC_VAR_BOOL | VLC_VAR_DOINHERIT );
81 p_playlist->p_input = NULL;
82 p_playlist->i_status = PLAYLIST_STOPPED;
83 p_playlist->i_index = -1;
84 p_playlist->i_size = 0;
85 p_playlist->pp_items = NULL;
87 if( vlc_thread_create( p_playlist, "playlist", RunThread,
88 VLC_THREAD_PRIORITY_LOW, VLC_TRUE ) )
90 msg_Err( p_playlist, "cannot spawn playlist thread" );
91 vlc_object_destroy( p_playlist );
95 /* The object has been initialized, now attach it */
96 vlc_object_attach( p_playlist, p_parent );
102 * Destroy the playlist.
104 * Delete all items in the playlist and free the playlist structure.
105 * \param p_playlist the playlist structure to destroy
107 void playlist_Destroy( playlist_t * p_playlist )
109 p_playlist->b_die = 1;
111 vlc_thread_join( p_playlist );
113 var_Destroy( p_playlist, "intf-change" );
115 vlc_object_destroy( p_playlist );
119 * Add an MRL to the playlist. This is a simplified version of
120 * playlist_AddExt inculded for convenince. It equals calling playlist_AddExt
121 * with psz_name == psz_target and i_duration == -1
124 int playlist_Add( playlist_t *p_playlist, const char *psz_target,
125 const char **ppsz_options, int i_options,
126 int i_mode, int i_pos )
128 return playlist_AddExt( p_playlist, psz_target, psz_target, -1,
129 ppsz_options, i_options, i_mode, i_pos );
133 * Add a MRL into the playlist.
135 * \param p_playlist the playlist to add into
136 * \param psz_uri the mrl to add to the playlist
137 * \param psz_name a text giving a name or description of this item
138 * \param i_duration a hint about the duration of this item, in miliseconds, or
140 * \param ppsz_options array of options
141 * \param i_options number of items in ppsz_options
142 * \param i_mode the mode used when adding
143 * \param i_pos the position in the playlist where to add. If this is
144 * PLAYLIST_END the item will be added at the end of the playlist
145 * regardless of it's size
146 * \return always returns 0
148 int playlist_AddExt( playlist_t *p_playlist, const char * psz_uri,
149 const char * psz_name, mtime_t i_duration,
150 const char **ppsz_options, int i_options, int i_mode,
153 playlist_item_t * p_item;
155 p_item = malloc( sizeof( playlist_item_t ) );
158 msg_Err( p_playlist, "out of memory" );
161 p_item->psz_name = strdup( psz_name );
162 p_item->psz_uri = strdup( psz_uri );
163 p_item->i_duration = i_duration;
165 p_item->i_status = 0;
166 p_item->b_autodeletion = VLC_FALSE;
168 p_item->ppsz_options = NULL;
169 p_item->i_options = i_options;
175 p_item->ppsz_options = (char **)malloc( i_options * sizeof(char *) );
176 for( i = 0; i < i_options; i++ )
177 p_item->ppsz_options[i] = strdup( ppsz_options[i] );
181 return playlist_AddItem( p_playlist, p_item, i_mode, i_pos );
185 * Add a playlist item into a playlist
187 * \param p_playlist the playlist to insert into
188 * \param p_item the playlist item to insert
189 * \param i_mode the mode used when adding
190 * \param i_pos the possition in the playlist where to add. If this is
191 * PLAYLIST_END the item will be added at the end of the playlist
192 * regardless of it's size
193 * \return always returns 0
195 int playlist_AddItem( playlist_t *p_playlist, playlist_item_t * p_item,
196 int i_mode, int i_pos)
200 vlc_mutex_lock( &p_playlist->object_lock );
203 * CHECK_INSERT : checks if the item is already enqued before
206 if ( i_mode & PLAYLIST_CHECK_INSERT )
210 if ( p_playlist->pp_items )
212 for ( j = 0; j < p_playlist->i_size; j++ )
214 if ( !strcmp( p_playlist->pp_items[j]->psz_uri, p_item->psz_uri ) )
216 if( p_item->psz_name )
218 free( p_item->psz_name );
220 if( p_item->psz_uri )
222 free( p_item->psz_uri );
225 vlc_mutex_unlock( &p_playlist->object_lock );
230 i_mode &= ~PLAYLIST_CHECK_INSERT;
231 i_mode |= PLAYLIST_APPEND;
235 msg_Dbg( p_playlist, "adding playlist item « %s » ( %s )", p_item->psz_name, p_item->psz_uri);
237 /* Create the new playlist item */
240 /* Do a few boundary checks and allocate space for the item */
241 if( i_pos == PLAYLIST_END )
243 if( i_mode & PLAYLIST_INSERT )
245 i_mode &= ~PLAYLIST_INSERT;
246 i_mode |= PLAYLIST_APPEND;
249 i_pos = p_playlist->i_size - 1;
252 if( !(i_mode & PLAYLIST_REPLACE)
253 || i_pos < 0 || i_pos >= p_playlist->i_size )
255 /* Additional boundary checks */
256 if( i_mode & PLAYLIST_APPEND )
265 else if( i_pos > p_playlist->i_size )
267 i_pos = p_playlist->i_size;
270 INSERT_ELEM( p_playlist->pp_items,
275 if( p_playlist->i_index >= i_pos )
277 p_playlist->i_index++;
282 /* i_mode == PLAYLIST_REPLACE and 0 <= i_pos < p_playlist->i_size */
283 if( p_playlist->pp_items[i_pos]->psz_name )
285 free( p_playlist->pp_items[i_pos]->psz_name );
287 if( p_playlist->pp_items[i_pos]->psz_uri )
289 free( p_playlist->pp_items[i_pos]->psz_uri );
291 /* XXX: what if the item is still in use? */
292 free( p_playlist->pp_items[i_pos] );
293 p_playlist->pp_items[i_pos] = p_item;
296 if( i_mode & PLAYLIST_GO )
298 p_playlist->i_index = i_pos;
299 if( p_playlist->p_input )
301 input_StopThread( p_playlist->p_input );
303 p_playlist->i_status = PLAYLIST_RUNNING;
306 vlc_mutex_unlock( &p_playlist->object_lock );
308 val.b_bool = VLC_TRUE;
309 var_Set( p_playlist, "intf-change", val );
315 * delete an item from a playlist.
317 * \param p_playlist the playlist to remove from.
318 * \param i_pos the position of the item to remove
321 int playlist_Delete( playlist_t * p_playlist, int i_pos )
324 vlc_mutex_lock( &p_playlist->object_lock );
326 if( i_pos >= 0 && i_pos < p_playlist->i_size )
328 msg_Dbg( p_playlist, "deleting playlist item « %s »",
329 p_playlist->pp_items[i_pos]->psz_name );
331 if( p_playlist->pp_items[i_pos]->psz_name )
333 free( p_playlist->pp_items[i_pos]->psz_name );
335 if( p_playlist->pp_items[i_pos]->psz_uri )
337 free( p_playlist->pp_items[i_pos]->psz_uri );
339 if( p_playlist->pp_items[i_pos]->i_options )
343 for( i = 0; i < p_playlist->pp_items[i_pos]->i_options; i++ )
344 free( p_playlist->pp_items[i_pos]->ppsz_options[i] );
346 free( p_playlist->pp_items[i_pos]->ppsz_options );
349 /* XXX: what if the item is still in use? */
350 free( p_playlist->pp_items[i_pos] );
352 if( i_pos <= p_playlist->i_index )
354 p_playlist->i_index--;
357 /* Renumber the playlist */
358 REMOVE_ELEM( p_playlist->pp_items,
363 vlc_mutex_unlock( &p_playlist->object_lock );
365 val.b_bool = VLC_TRUE;
366 var_Set( p_playlist, "intf-change", val );
376 int playlist_Sort( playlist_t * p_playlist , int i_type )
378 int i , i_small , i_position;
379 playlist_item_t *p_temp;
381 vlc_mutex_lock( &p_playlist->object_lock );
383 for( i_position = 0; i_position < p_playlist->i_size -1 ; i_position ++ )
385 i_small = i_position;
386 for( i = i_position + 1 ; i< p_playlist->i_size ; i++)
390 i_test = strcasecmp( p_playlist->pp_items[i]->psz_name,
391 p_playlist->pp_items[i_small]->psz_name );
393 if( ( i_type == SORT_NORMAL && i_test < 0 ) ||
394 ( i_type == SORT_REVERSE && i_test > 0 ) )
399 p_temp = p_playlist->pp_items[i_position];
400 p_playlist->pp_items[i_position] = p_playlist->pp_items[i_small];
401 p_playlist->pp_items[i_small] = p_temp;
403 vlc_mutex_unlock( &p_playlist->object_lock );
409 * Move an item in a playlist
411 * Move the item in the playlist with position i_pos before the current item
412 * at position i_newpos.
413 * \param p_playlist the playlist to move items in
414 * \param i_pos the position of the item to move
415 * \param i_newpos the position of the item that will be behind the moved item
419 int playlist_Move( playlist_t * p_playlist, int i_pos, int i_newpos)
422 vlc_mutex_lock( &p_playlist->object_lock );
424 /* take into account that our own row disappears. */
425 if ( i_pos < i_newpos ) i_newpos--;
427 if( i_pos >= 0 && i_newpos >=0 && i_pos <= p_playlist->i_size
428 && i_newpos <= p_playlist->i_size )
430 playlist_item_t * temp;
432 msg_Dbg( p_playlist, "moving playlist item « %s »",
433 p_playlist->pp_items[i_pos]->psz_name );
435 if( i_pos == p_playlist->i_index )
437 p_playlist->i_index = i_newpos;
439 else if( i_pos > p_playlist->i_index && i_newpos <= p_playlist->i_index )
441 p_playlist->i_index++;
443 else if( i_pos < p_playlist->i_index && i_newpos >= p_playlist->i_index )
445 p_playlist->i_index--;
448 if ( i_pos < i_newpos )
450 temp = p_playlist->pp_items[i_pos];
451 while ( i_pos < i_newpos )
453 p_playlist->pp_items[i_pos] = p_playlist->pp_items[i_pos+1];
456 p_playlist->pp_items[i_newpos] = temp;
458 else if ( i_pos > i_newpos )
460 temp = p_playlist->pp_items[i_pos];
461 while ( i_pos > i_newpos )
463 p_playlist->pp_items[i_pos] = p_playlist->pp_items[i_pos-1];
466 p_playlist->pp_items[i_newpos] = temp;
470 vlc_mutex_unlock( &p_playlist->object_lock );
472 val.b_bool = VLC_TRUE;
473 var_Set( p_playlist, "intf-change", val );
479 * Do a playlist action
481 * \param p_playlist the playlist to do the command on
482 * \param i_command the command to do
483 * \param i_arg the argument to the command. See playlist_command_t for details
485 void playlist_Command( playlist_t * p_playlist, playlist_command_t i_command,
490 vlc_mutex_lock( &p_playlist->object_lock );
495 p_playlist->i_status = PLAYLIST_STOPPED;
496 if( p_playlist->p_input )
498 input_StopThread( p_playlist->p_input );
503 p_playlist->i_status = PLAYLIST_RUNNING;
504 if( !p_playlist->p_input )
506 PlayItem( p_playlist );
508 if( p_playlist->p_input )
510 val.i_int = PLAYING_S;
511 var_Set( p_playlist->p_input, "state", val );
516 p_playlist->i_status = PLAYLIST_PAUSED;
517 if( p_playlist->p_input )
520 var_Set( p_playlist->p_input, "state", val );
525 p_playlist->i_status = PLAYLIST_STOPPED;
526 SkipItem( p_playlist, i_arg );
527 if( p_playlist->p_input )
529 input_StopThread( p_playlist->p_input );
531 p_playlist->i_status = PLAYLIST_RUNNING;
535 if( i_arg >= 0 && i_arg < p_playlist->i_size )
537 p_playlist->i_index = i_arg;
538 if( p_playlist->p_input )
540 input_StopThread( p_playlist->p_input );
542 p_playlist->i_status = PLAYLIST_RUNNING;
547 msg_Err( p_playlist, "unknown playlist command" );
551 vlc_mutex_unlock( &p_playlist->object_lock );
555 /* Following functions are local */
557 static void ObjectGarbageCollector( playlist_t *p_playlist,
559 vlc_bool_t *pb_obj_destroyed,
560 mtime_t *pi_obj_destroyed_date )
563 if( *pb_obj_destroyed || *pi_obj_destroyed_date > mdate() )
568 if( *pi_obj_destroyed_date == 0 )
570 /* give a little time */
571 *pi_obj_destroyed_date = mdate() + 300000LL;
575 while( ( p_obj = vlc_object_find( p_playlist,
579 if( p_obj->p_parent != (vlc_object_t*)p_playlist )
581 /* only first chiled (ie unused) */
582 vlc_object_release( p_obj );
585 if( i_type == VLC_OBJECT_VOUT )
587 msg_Dbg( p_playlist, "vout garbage collector destroying 1 vout" );
588 vlc_object_detach( p_obj );
589 vlc_object_release( p_obj );
590 vout_Destroy( (vout_thread_t *)p_obj );
592 else if( i_type == VLC_OBJECT_SOUT )
594 vlc_object_release( p_obj );
595 sout_DeleteInstance( (sout_instance_t*)p_obj );
598 *pb_obj_destroyed = VLC_TRUE;
602 /*****************************************************************************
603 * RunThread: main playlist thread
604 *****************************************************************************/
605 static void RunThread ( playlist_t *p_playlist )
608 vlc_bool_t b_vout_destroyed = VLC_FALSE; /*we do vout garbage collector */
609 mtime_t i_vout_destroyed_date = 0;
611 vlc_bool_t b_sout_destroyed = VLC_FALSE; /*we do vout garbage collector */
612 mtime_t i_sout_destroyed_date = 0;
614 /* Tell above that we're ready */
615 vlc_thread_ready( p_playlist );
617 while( !p_playlist->b_die )
619 vlc_mutex_lock( &p_playlist->object_lock );
621 /* If there is an input, check that it doesn't need to die. */
622 if( p_playlist->p_input )
624 /* This input is dead. Remove it ! */
625 if( p_playlist->p_input->b_dead )
627 input_thread_t *p_input;
629 p_input = p_playlist->p_input;
630 p_playlist->p_input = NULL;
632 /* Release the playlist lock, because we may get stuck
633 * in input_DestroyThread() for some time. */
634 vlc_mutex_unlock( &p_playlist->object_lock );
637 input_DestroyThread( p_input );
639 /* Unlink current input (_after_ input_DestroyThread for vout garbage collector)*/
640 vlc_object_detach( p_input );
643 vlc_object_destroy( p_input );
645 b_vout_destroyed = VLC_FALSE;
646 i_vout_destroyed_date = 0;
647 b_sout_destroyed = VLC_FALSE;
648 i_sout_destroyed_date = 0;
651 /* This input is dying, let him do */
652 else if( p_playlist->p_input->b_die )
656 /* This input has finished, ask him to die ! */
657 else if( p_playlist->p_input->b_error
658 || p_playlist->p_input->b_eof )
660 /* Check for autodeletion */
661 if( p_playlist->pp_items[p_playlist->i_index]->b_autodeletion )
663 vlc_mutex_unlock( &p_playlist->object_lock );
664 playlist_Delete( p_playlist, p_playlist->i_index );
665 vlc_mutex_lock( &p_playlist->object_lock );
668 /* Select the next playlist item */
669 SkipItem( p_playlist, 1 );
671 input_StopThread( p_playlist->p_input );
672 vlc_mutex_unlock( &p_playlist->object_lock );
675 else if( p_playlist->p_input->stream.control.i_status != INIT_S )
677 vlc_mutex_unlock( &p_playlist->object_lock );
678 ObjectGarbageCollector( p_playlist, VLC_OBJECT_VOUT,
680 &i_vout_destroyed_date );
681 ObjectGarbageCollector( p_playlist, VLC_OBJECT_SOUT,
683 &i_sout_destroyed_date );
684 vlc_mutex_lock( &p_playlist->object_lock );
687 else if( p_playlist->i_status != PLAYLIST_STOPPED )
689 SkipItem( p_playlist, 0 );
690 PlayItem( p_playlist );
692 else if( p_playlist->i_status == PLAYLIST_STOPPED )
694 vlc_mutex_unlock( &p_playlist->object_lock );
695 ObjectGarbageCollector( p_playlist, VLC_OBJECT_SOUT,
696 &b_sout_destroyed, &i_sout_destroyed_date );
697 ObjectGarbageCollector( p_playlist, VLC_OBJECT_VOUT,
698 &b_vout_destroyed, &i_vout_destroyed_date );
699 vlc_mutex_lock( &p_playlist->object_lock );
701 vlc_mutex_unlock( &p_playlist->object_lock );
703 msleep( INTF_IDLE_SLEEP );
706 /* If there is an input, kill it */
709 vlc_mutex_lock( &p_playlist->object_lock );
711 if( p_playlist->p_input == NULL )
713 vlc_mutex_unlock( &p_playlist->object_lock );
717 if( p_playlist->p_input->b_dead )
719 input_thread_t *p_input;
721 /* Unlink current input */
722 p_input = p_playlist->p_input;
723 p_playlist->p_input = NULL;
724 vlc_mutex_unlock( &p_playlist->object_lock );
727 input_DestroyThread( p_input );
728 /* Unlink current input (_after_ input_DestroyThread for vout
729 * garbage collector)*/
730 vlc_object_detach( p_input );
733 vlc_object_destroy( p_input );
736 else if( p_playlist->p_input->b_die )
738 /* This input is dying, leave him alone */
741 else if( p_playlist->p_input->b_error || p_playlist->p_input->b_eof )
743 input_StopThread( p_playlist->p_input );
744 vlc_mutex_unlock( &p_playlist->object_lock );
749 p_playlist->p_input->b_eof = 1;
752 vlc_mutex_unlock( &p_playlist->object_lock );
754 msleep( INTF_IDLE_SLEEP );
757 /* close all remaining sout */
758 while( ( p_obj = vlc_object_find( p_playlist,
759 VLC_OBJECT_SOUT, FIND_CHILD ) ) )
761 vlc_object_release( p_obj );
762 sout_DeleteInstance( (sout_instance_t*)p_obj );
765 /* close all remaining vout */
766 while( ( p_obj = vlc_object_find( p_playlist,
767 VLC_OBJECT_VOUT, FIND_CHILD ) ) )
769 vlc_object_detach( p_obj );
770 vlc_object_release( p_obj );
771 vout_Destroy( (vout_thread_t *)p_obj );
775 /*****************************************************************************
776 * SkipItem: go to Xth playlist item
777 *****************************************************************************
778 * This function calculates the position of the next playlist item, depending
779 * on the playlist course mode (forward, backward, random...).
780 *****************************************************************************/
781 static void SkipItem( playlist_t *p_playlist, int i_arg )
783 int i_oldindex = p_playlist->i_index;
784 vlc_bool_t b_random, b_repeat, b_loop;
787 /* If the playlist is empty, there is no current item */
788 if( p_playlist->i_size == 0 )
790 p_playlist->i_index = -1;
794 var_Get( p_playlist, "random", &val );
795 b_random = val.b_bool;
796 var_Get( p_playlist, "repeat", &val );
797 b_repeat = val.b_bool;
798 var_Get( p_playlist, "loop", &val );
804 srand( (unsigned int)mdate() );
806 /* Simple random stuff - we cheat a bit to minimize the chances to
807 * get the same index again. */
808 i_arg = (int)((float)p_playlist->i_size * rand() / (RAND_MAX+1.0));
811 i_arg = (int)((float)p_playlist->i_size * rand() / (RAND_MAX+1.0));
818 p_playlist->i_index += i_arg;
821 if( p_playlist->i_index >= p_playlist->i_size )
823 if( p_playlist->i_status == PLAYLIST_STOPPED
827 p_playlist->i_index -= p_playlist->i_size
828 * ( p_playlist->i_index / p_playlist->i_size );
832 /* Don't loop by default: stop at playlist end */
833 p_playlist->i_index = i_oldindex;
834 p_playlist->i_status = PLAYLIST_STOPPED;
837 else if( p_playlist->i_index < 0 )
839 p_playlist->i_index = p_playlist->i_size - 1;
842 val.b_bool = VLC_TRUE;
843 var_Set( p_playlist, "intf-change", val );
846 /*****************************************************************************
847 * PlayItem: play current playlist item
848 *****************************************************************************
849 * This function calculates the position of the next playlist item, depending
850 * on the playlist course mode (forward, backward, random...).
851 *****************************************************************************/
852 static void PlayItem( playlist_t *p_playlist )
854 if( p_playlist->i_index == -1 )
856 if( p_playlist->i_size == 0 )
861 SkipItem( p_playlist, 1 );
864 msg_Dbg( p_playlist, "creating new input thread" );
865 p_playlist->p_input = input_CreateThread( p_playlist,
866 p_playlist->pp_items[p_playlist->i_index] );
869 /*****************************************************************************
870 * Poubellize: put an input thread in the trashcan
871 *****************************************************************************
873 *****************************************************************************/
874 static void Poubellize ( playlist_t *p_playlist, input_thread_t *p_input )
876 msg_Dbg( p_playlist, "poubellizing input %i\n", p_input->i_object_id );
879 /*****************************************************************************
880 * playlist_LoadFile: load a playlist file.
881 ****************************************************************************/
882 int playlist_LoadFile( playlist_t * p_playlist, const char *psz_filename )
886 int i_current_status;
889 msg_Dbg( p_playlist, "opening playlist file %s", psz_filename );
891 file = fopen( psz_filename, "rt" );
894 msg_Err( p_playlist, "playlist file %s does not exist", psz_filename );
897 fseek( file, 0L, SEEK_SET );
899 /* check the file is not empty */
900 if ( ! fgets( line, 1024, file ) )
902 msg_Err( p_playlist, "playlist file %s is empty", psz_filename );
907 /* get rid of line feed */
908 if( line[strlen(line)-1] == '\n' || line[strlen(line)-1] == '\r' )
910 line[strlen(line)-1] = (char)0;
911 if( line[strlen(line)-1] == '\r' ) line[strlen(line)-1] = (char)0;
913 /* check the file format is valid */
914 if ( strcmp ( line , PLAYLIST_FILE_HEADER_0_5 ) )
916 msg_Err( p_playlist, "playlist file %s format is unsupported"
923 i_current_status = p_playlist->i_status;
924 if ( p_playlist->i_status != PLAYLIST_STOPPED )
926 playlist_Stop ( p_playlist );
929 /* delete current content of the playlist */
930 for( i = p_playlist->i_size - 1; i >= 0; i-- )
932 playlist_Delete ( p_playlist , i );
935 /* simply add each line */
936 while( fgets( line, 1024, file ) )
938 /* ignore comments or empty lines */
939 if( (line[0] == '#') || (line[0] == '\r') || (line[0] == '\n')
940 || (line[0] == (char)0) )
943 /* get rid of line feed */
944 if( line[strlen(line)-1] == '\n' || line[strlen(line)-1] == '\r' )
946 line[strlen(line)-1] = (char)0;
947 if( line[strlen(line)-1] == '\r' ) line[strlen(line)-1] = (char)0;
950 playlist_Add ( p_playlist , (char *)&line ,
951 0, 0, PLAYLIST_APPEND , PLAYLIST_END );
955 if ( i_current_status != PLAYLIST_STOPPED )
957 playlist_Play ( p_playlist );
965 /*****************************************************************************
966 * playlist_SaveFile: Save a playlist in a file.
967 *****************************************************************************/
968 int playlist_SaveFile( playlist_t * p_playlist, const char * psz_filename )
973 vlc_mutex_lock( &p_playlist->object_lock );
975 msg_Dbg( p_playlist, "saving playlist file %s", psz_filename );
977 file = fopen( psz_filename, "wt" );
980 msg_Err( p_playlist , "could not create playlist file %s"
985 fprintf( file , PLAYLIST_FILE_HEADER_0_5 "\n" );
987 for ( i = 0 ; i < p_playlist->i_size ; i++ )
989 fprintf( file , p_playlist->pp_items[i]->psz_uri );
990 fprintf( file , "\n" );
995 vlc_mutex_unlock( &p_playlist->object_lock );