1 /*****************************************************************************
2 * playlist.c : Playlist management functions
3 *****************************************************************************
4 * Copyright (C) 1999-2001 VideoLAN
5 * $Id: playlist.c,v 1.54 2003/09/20 19:37:54 hartman 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;
405 for( i=0; i < p_playlist->i_size; i++ )
407 msg_Dbg( p_playlist, "%s", p_playlist->pp_items[i]->psz_name );
410 vlc_mutex_unlock( &p_playlist->object_lock );
416 * Move an item in a playlist
418 * Move the item in the playlist with position i_pos before the current item
419 * at position i_newpos.
420 * \param p_playlist the playlist to move items in
421 * \param i_pos the position of the item to move
422 * \param i_newpos the position of the item that will be behind the moved item
426 int playlist_Move( playlist_t * p_playlist, int i_pos, int i_newpos)
429 vlc_mutex_lock( &p_playlist->object_lock );
431 /* take into account that our own row disappears. */
432 if ( i_pos < i_newpos ) i_newpos--;
434 if( i_pos >= 0 && i_newpos >=0 && i_pos <= p_playlist->i_size
435 && i_newpos <= p_playlist->i_size )
437 playlist_item_t * temp;
439 msg_Dbg( p_playlist, "moving playlist item « %s »",
440 p_playlist->pp_items[i_pos]->psz_name );
442 if( i_pos == p_playlist->i_index )
444 p_playlist->i_index = i_newpos;
446 else if( i_pos > p_playlist->i_index && i_newpos <= p_playlist->i_index )
448 p_playlist->i_index++;
450 else if( i_pos < p_playlist->i_index && i_newpos >= p_playlist->i_index )
452 p_playlist->i_index--;
455 if ( i_pos < i_newpos )
457 temp = p_playlist->pp_items[i_pos];
458 while ( i_pos < i_newpos )
460 p_playlist->pp_items[i_pos] = p_playlist->pp_items[i_pos+1];
463 p_playlist->pp_items[i_newpos] = temp;
465 else if ( i_pos > i_newpos )
467 temp = p_playlist->pp_items[i_pos];
468 while ( i_pos > i_newpos )
470 p_playlist->pp_items[i_pos] = p_playlist->pp_items[i_pos-1];
473 p_playlist->pp_items[i_newpos] = temp;
477 vlc_mutex_unlock( &p_playlist->object_lock );
479 val.b_bool = VLC_TRUE;
480 var_Set( p_playlist, "intf-change", val );
486 * Do a playlist action
488 * \param p_playlist the playlist to do the command on
489 * \param i_command the command to do
490 * \param i_arg the argument to the command. See playlist_command_t for details
492 void playlist_Command( playlist_t * p_playlist, playlist_command_t i_command,
497 vlc_mutex_lock( &p_playlist->object_lock );
502 p_playlist->i_status = PLAYLIST_STOPPED;
503 if( p_playlist->p_input )
505 input_StopThread( p_playlist->p_input );
510 p_playlist->i_status = PLAYLIST_RUNNING;
511 if( !p_playlist->p_input )
513 PlayItem( p_playlist );
515 if( p_playlist->p_input )
517 val.i_int = PLAYING_S;
518 var_Set( p_playlist->p_input, "state", val );
523 p_playlist->i_status = PLAYLIST_PAUSED;
524 if( p_playlist->p_input )
527 var_Set( p_playlist->p_input, "state", val );
532 p_playlist->i_status = PLAYLIST_STOPPED;
533 SkipItem( p_playlist, i_arg );
534 if( p_playlist->p_input )
536 input_StopThread( p_playlist->p_input );
538 p_playlist->i_status = PLAYLIST_RUNNING;
542 if( i_arg >= 0 && i_arg < p_playlist->i_size )
544 p_playlist->i_index = i_arg;
545 if( p_playlist->p_input )
547 input_StopThread( p_playlist->p_input );
549 p_playlist->i_status = PLAYLIST_RUNNING;
554 msg_Err( p_playlist, "unknown playlist command" );
558 vlc_mutex_unlock( &p_playlist->object_lock );
562 /* Following functions are local */
564 static void ObjectGarbageCollector( playlist_t *p_playlist,
566 vlc_bool_t *pb_obj_destroyed,
567 mtime_t *pi_obj_destroyed_date )
570 if( *pb_obj_destroyed || *pi_obj_destroyed_date > mdate() )
575 if( *pi_obj_destroyed_date == 0 )
577 /* give a little time */
578 *pi_obj_destroyed_date = mdate() + 300000LL;
582 while( ( p_obj = vlc_object_find( p_playlist,
586 if( p_obj->p_parent != (vlc_object_t*)p_playlist )
588 /* only first chiled (ie unused) */
589 vlc_object_release( p_obj );
592 if( i_type == VLC_OBJECT_VOUT )
594 msg_Dbg( p_playlist, "vout garbage collector destroying 1 vout" );
595 vlc_object_detach( p_obj );
596 vlc_object_release( p_obj );
597 vout_Destroy( (vout_thread_t *)p_obj );
599 else if( i_type == VLC_OBJECT_SOUT )
601 vlc_object_release( p_obj );
602 sout_DeleteInstance( (sout_instance_t*)p_obj );
605 *pb_obj_destroyed = VLC_TRUE;
609 /*****************************************************************************
610 * RunThread: main playlist thread
611 *****************************************************************************/
612 static void RunThread ( playlist_t *p_playlist )
615 vlc_bool_t b_vout_destroyed = VLC_FALSE; /*we do vout garbage collector */
616 mtime_t i_vout_destroyed_date = 0;
618 vlc_bool_t b_sout_destroyed = VLC_FALSE; /*we do vout garbage collector */
619 mtime_t i_sout_destroyed_date = 0;
621 /* Tell above that we're ready */
622 vlc_thread_ready( p_playlist );
624 while( !p_playlist->b_die )
626 vlc_mutex_lock( &p_playlist->object_lock );
628 /* If there is an input, check that it doesn't need to die. */
629 if( p_playlist->p_input )
631 /* This input is dead. Remove it ! */
632 if( p_playlist->p_input->b_dead )
634 input_thread_t *p_input;
636 p_input = p_playlist->p_input;
637 p_playlist->p_input = NULL;
639 /* Release the playlist lock, because we may get stuck
640 * in input_DestroyThread() for some time. */
641 vlc_mutex_unlock( &p_playlist->object_lock );
644 input_DestroyThread( p_input );
646 /* Unlink current input (_after_ input_DestroyThread for vout garbage collector)*/
647 vlc_object_detach( p_input );
650 vlc_object_destroy( p_input );
652 b_vout_destroyed = VLC_FALSE;
653 i_vout_destroyed_date = 0;
654 b_sout_destroyed = VLC_FALSE;
655 i_sout_destroyed_date = 0;
658 /* This input is dying, let him do */
659 else if( p_playlist->p_input->b_die )
663 /* This input has finished, ask him to die ! */
664 else if( p_playlist->p_input->b_error
665 || p_playlist->p_input->b_eof )
667 /* Check for autodeletion */
668 if( p_playlist->pp_items[p_playlist->i_index]->b_autodeletion )
670 vlc_mutex_unlock( &p_playlist->object_lock );
671 playlist_Delete( p_playlist, p_playlist->i_index );
672 vlc_mutex_lock( &p_playlist->object_lock );
675 /* Select the next playlist item */
676 SkipItem( p_playlist, 1 );
678 input_StopThread( p_playlist->p_input );
679 vlc_mutex_unlock( &p_playlist->object_lock );
682 else if( p_playlist->p_input->stream.control.i_status != INIT_S )
684 vlc_mutex_unlock( &p_playlist->object_lock );
685 ObjectGarbageCollector( p_playlist, VLC_OBJECT_VOUT,
687 &i_vout_destroyed_date );
688 ObjectGarbageCollector( p_playlist, VLC_OBJECT_SOUT,
690 &i_sout_destroyed_date );
691 vlc_mutex_lock( &p_playlist->object_lock );
694 else if( p_playlist->i_status != PLAYLIST_STOPPED )
696 SkipItem( p_playlist, 0 );
697 PlayItem( p_playlist );
699 else if( p_playlist->i_status == PLAYLIST_STOPPED )
701 vlc_mutex_unlock( &p_playlist->object_lock );
702 ObjectGarbageCollector( p_playlist, VLC_OBJECT_SOUT,
703 &b_sout_destroyed, &i_sout_destroyed_date );
704 ObjectGarbageCollector( p_playlist, VLC_OBJECT_VOUT,
705 &b_vout_destroyed, &i_vout_destroyed_date );
706 vlc_mutex_lock( &p_playlist->object_lock );
708 vlc_mutex_unlock( &p_playlist->object_lock );
710 msleep( INTF_IDLE_SLEEP );
713 /* If there is an input, kill it */
716 vlc_mutex_lock( &p_playlist->object_lock );
718 if( p_playlist->p_input == NULL )
720 vlc_mutex_unlock( &p_playlist->object_lock );
724 if( p_playlist->p_input->b_dead )
726 input_thread_t *p_input;
728 /* Unlink current input */
729 p_input = p_playlist->p_input;
730 p_playlist->p_input = NULL;
731 vlc_mutex_unlock( &p_playlist->object_lock );
734 input_DestroyThread( p_input );
735 /* Unlink current input (_after_ input_DestroyThread for vout
736 * garbage collector)*/
737 vlc_object_detach( p_input );
740 vlc_object_destroy( p_input );
743 else if( p_playlist->p_input->b_die )
745 /* This input is dying, leave him alone */
748 else if( p_playlist->p_input->b_error || p_playlist->p_input->b_eof )
750 input_StopThread( p_playlist->p_input );
751 vlc_mutex_unlock( &p_playlist->object_lock );
756 p_playlist->p_input->b_eof = 1;
759 vlc_mutex_unlock( &p_playlist->object_lock );
761 msleep( INTF_IDLE_SLEEP );
764 /* close all remaining sout */
765 while( ( p_obj = vlc_object_find( p_playlist,
766 VLC_OBJECT_SOUT, FIND_CHILD ) ) )
768 vlc_object_release( p_obj );
769 sout_DeleteInstance( (sout_instance_t*)p_obj );
772 /* close all remaining vout */
773 while( ( p_obj = vlc_object_find( p_playlist,
774 VLC_OBJECT_VOUT, FIND_CHILD ) ) )
776 vlc_object_detach( p_obj );
777 vlc_object_release( p_obj );
778 vout_Destroy( (vout_thread_t *)p_obj );
782 /*****************************************************************************
783 * SkipItem: go to Xth playlist item
784 *****************************************************************************
785 * This function calculates the position of the next playlist item, depending
786 * on the playlist course mode (forward, backward, random...).
787 *****************************************************************************/
788 static void SkipItem( playlist_t *p_playlist, int i_arg )
790 int i_oldindex = p_playlist->i_index;
791 vlc_bool_t b_random, b_repeat, b_loop;
794 /* If the playlist is empty, there is no current item */
795 if( p_playlist->i_size == 0 )
797 p_playlist->i_index = -1;
801 var_Get( p_playlist, "random", &val );
802 b_random = val.b_bool;
803 var_Get( p_playlist, "repeat", &val );
804 b_repeat = val.b_bool;
805 var_Get( p_playlist, "loop", &val );
811 srand( (unsigned int)mdate() );
813 /* Simple random stuff - we cheat a bit to minimize the chances to
814 * get the same index again. */
815 i_arg = (int)((float)p_playlist->i_size * rand() / (RAND_MAX+1.0));
818 i_arg = (int)((float)p_playlist->i_size * rand() / (RAND_MAX+1.0));
825 p_playlist->i_index += i_arg;
828 if( p_playlist->i_index >= p_playlist->i_size )
830 if( p_playlist->i_status == PLAYLIST_STOPPED
834 p_playlist->i_index -= p_playlist->i_size
835 * ( p_playlist->i_index / p_playlist->i_size );
839 /* Don't loop by default: stop at playlist end */
840 p_playlist->i_index = i_oldindex;
841 p_playlist->i_status = PLAYLIST_STOPPED;
844 else if( p_playlist->i_index < 0 )
846 p_playlist->i_index = p_playlist->i_size - 1;
849 val.b_bool = VLC_TRUE;
850 var_Set( p_playlist, "intf-change", val );
853 /*****************************************************************************
854 * PlayItem: play current playlist item
855 *****************************************************************************
856 * This function calculates the position of the next playlist item, depending
857 * on the playlist course mode (forward, backward, random...).
858 *****************************************************************************/
859 static void PlayItem( playlist_t *p_playlist )
861 if( p_playlist->i_index == -1 )
863 if( p_playlist->i_size == 0 )
868 SkipItem( p_playlist, 1 );
871 msg_Dbg( p_playlist, "creating new input thread" );
872 p_playlist->p_input = input_CreateThread( p_playlist,
873 p_playlist->pp_items[p_playlist->i_index] );
876 /*****************************************************************************
877 * Poubellize: put an input thread in the trashcan
878 *****************************************************************************
880 *****************************************************************************/
881 static void Poubellize ( playlist_t *p_playlist, input_thread_t *p_input )
883 msg_Dbg( p_playlist, "poubellizing input %i\n", p_input->i_object_id );
886 /*****************************************************************************
887 * playlist_LoadFile: load a playlist file.
888 ****************************************************************************/
889 int playlist_LoadFile( playlist_t * p_playlist, const char *psz_filename )
893 int i_current_status;
896 msg_Dbg( p_playlist, "opening playlist file %s", psz_filename );
898 file = fopen( psz_filename, "rt" );
901 msg_Err( p_playlist, "playlist file %s does not exist", psz_filename );
904 fseek( file, 0L, SEEK_SET );
906 /* check the file is not empty */
907 if ( ! fgets( line, 1024, file ) )
909 msg_Err( p_playlist, "playlist file %s is empty", psz_filename );
914 /* get rid of line feed */
915 if( line[strlen(line)-1] == '\n' || line[strlen(line)-1] == '\r' )
917 line[strlen(line)-1] = (char)0;
918 if( line[strlen(line)-1] == '\r' ) line[strlen(line)-1] = (char)0;
920 /* check the file format is valid */
921 if ( strcmp ( line , PLAYLIST_FILE_HEADER_0_5 ) )
923 msg_Err( p_playlist, "playlist file %s format is unsupported"
930 i_current_status = p_playlist->i_status;
931 if ( p_playlist->i_status != PLAYLIST_STOPPED )
933 playlist_Stop ( p_playlist );
936 /* delete current content of the playlist */
937 for( i = p_playlist->i_size - 1; i >= 0; i-- )
939 playlist_Delete ( p_playlist , i );
942 /* simply add each line */
943 while( fgets( line, 1024, file ) )
945 /* ignore comments or empty lines */
946 if( (line[0] == '#') || (line[0] == '\r') || (line[0] == '\n')
947 || (line[0] == (char)0) )
950 /* get rid of line feed */
951 if( line[strlen(line)-1] == '\n' || line[strlen(line)-1] == '\r' )
953 line[strlen(line)-1] = (char)0;
954 if( line[strlen(line)-1] == '\r' ) line[strlen(line)-1] = (char)0;
957 playlist_Add ( p_playlist , (char *)&line ,
958 0, 0, PLAYLIST_APPEND , PLAYLIST_END );
962 if ( i_current_status != PLAYLIST_STOPPED )
964 playlist_Play ( p_playlist );
972 /*****************************************************************************
973 * playlist_SaveFile: Save a playlist in a file.
974 *****************************************************************************/
975 int playlist_SaveFile( playlist_t * p_playlist, const char * psz_filename )
980 vlc_mutex_lock( &p_playlist->object_lock );
982 msg_Dbg( p_playlist, "saving playlist file %s", psz_filename );
984 file = fopen( psz_filename, "wt" );
987 msg_Err( p_playlist , "could not create playlist file %s"
992 fprintf( file , PLAYLIST_FILE_HEADER_0_5 "\n" );
994 for ( i = 0 ; i < p_playlist->i_size ; i++ )
996 fprintf( file , p_playlist->pp_items[i]->psz_uri );
997 fprintf( file , "\n" );
1002 vlc_mutex_unlock( &p_playlist->object_lock );