]> git.sesse.net Git - vlc/blob - modules/demux/playlist/b4s.c
Fixed support of mov/mp4 with fixed video samples size.
[vlc] / modules / demux / playlist / b4s.c
1 /*****************************************************************************
2  * b4s.c : B4S playlist format import
3  *****************************************************************************
4  * Copyright (C) 2005-2009 the VideoLAN team
5  * $Id$
6  *
7  * Authors: Sigmund Augdal Helberg <dnumgis@videolan.org>
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., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
22  *****************************************************************************/
23
24 /*****************************************************************************
25  * Preamble
26  *****************************************************************************/
27
28 #ifdef HAVE_CONFIG_H
29 # include "config.h"
30 #endif
31
32 #include <vlc_common.h>
33 #include <vlc_demux.h>
34 #include <vlc_xml.h>
35
36 #include "playlist.h"
37
38 struct demux_sys_t
39 {
40 };
41
42 /*****************************************************************************
43  * Local prototypes
44  *****************************************************************************/
45 static int Demux( demux_t *p_demux);
46 static int Control( demux_t *p_demux, int i_query, va_list args );
47 static bool IsWhitespace( const char *psz_string );
48
49 /*****************************************************************************
50  * Import_B4S: main import function
51  *****************************************************************************/
52 int Import_B4S( vlc_object_t *p_this )
53 {
54     DEMUX_BY_EXTENSION_OR_FORCED_MSG( ".b4s", "b4s-open",
55                                       "using B4S playlist reader" );
56     return VLC_SUCCESS;
57 }
58
59 /*****************************************************************************
60  * Deactivate: frees unused data
61  *****************************************************************************/
62 void Close_B4S( vlc_object_t *p_this )
63 {
64     demux_t *p_demux = (demux_t *)p_this;
65     demux_sys_t *p_sys = p_demux->p_sys;
66
67     free( p_sys );
68 }
69
70 static int Demux( demux_t *p_demux )
71 {
72     int i_ret = -1;
73
74     xml_reader_t *p_xml_reader = NULL;
75     char *psz_elname = NULL;
76     const char *node;
77     input_item_t *p_input;
78     char *psz_mrl = NULL, *psz_title = NULL, *psz_genre = NULL;
79     char *psz_now = NULL, *psz_listeners = NULL, *psz_bitrate = NULL;
80     input_item_node_t *p_subitems = NULL;
81
82     input_item_t *p_current_input = GetCurrentItem(p_demux);
83
84     free( stream_ReadLine( p_demux->s ) );
85
86     p_xml_reader = xml_ReaderCreate( p_demux, p_demux->s );
87     if( !p_xml_reader )
88         return -1;
89
90     /* xml */
91     /* check root node */
92     if( xml_ReaderNextNode( p_xml_reader, &node ) != XML_READER_STARTELEM )
93     {
94         msg_Err( p_demux, "invalid file (no root node)" );
95         goto end;
96     }
97
98     if( strcmp( node, "WinampXML" ) )
99     {
100         msg_Err( p_demux, "invalid root node: %s", node );
101         goto end;
102     }
103
104     /* root node should not have any attributes, and should only
105      * contain the "playlist node */
106
107     /* Skip until 1st child node */
108     while( (i_ret = xml_ReaderNextNode( p_xml_reader, &node )) != XML_READER_STARTELEM )
109         if( i_ret <= 0 )
110         {
111             msg_Err( p_demux, "invalid file (no child node)" );
112             goto end;
113         }
114
115     if( strcmp( node, "playlist" ) )
116     {
117         msg_Err( p_demux, "invalid child node %s", node );
118         goto end;
119     }
120
121     // Read the attributes
122     const char *attr, *value;
123     while( (attr = xml_ReaderNextAttr( p_xml_reader, &value )) != NULL )
124     {
125         if( !strcmp( attr, "num_entries" ) )
126             msg_Dbg( p_demux, "playlist has %d entries", atoi(value) );
127         else if( !strcmp( attr, "label" ) )
128             input_item_SetName( p_current_input, value );
129         else
130             msg_Warn( p_demux, "stray attribute %s with value %s in element"
131                       " <playlist>", attr, value );
132     }
133
134     p_subitems = input_item_node_Create( p_current_input );
135
136     while( (i_ret = xml_ReaderNextNode( p_xml_reader, &node )) > 0 )
137     {
138         // Get the node type
139         switch( i_ret )
140         {
141             case XML_READER_STARTELEM:
142             {
143                 // Read the element name
144                 free( psz_elname );
145                 psz_elname = strdup( node );
146                 if( unlikely(!psz_elname) )
147                     goto end;
148
149                 // Read the attributes
150                 while( (attr = xml_ReaderNextAttr( p_xml_reader, &value )) )
151                 {
152                     if( !strcmp( psz_elname, "entry" ) &&
153                         !strcmp( attr, "Playstring" ) )
154                     {
155                         free( psz_mrl );
156                         psz_mrl = strdup( value );
157                     }
158                     else
159                     {
160                         msg_Warn( p_demux, "unexpected attribute %s in <%s>",
161                                   attr, psz_elname );
162                     }
163                 }
164                 break;
165             }
166
167             case XML_READER_TEXT:
168             {
169                 char **p;
170
171                 if( psz_elname == NULL )
172                     break;
173                 if( IsWhitespace( node ) )
174                     break;
175                 if( !strcmp( psz_elname, "Name" ) )
176                     p = &psz_title;
177                 else if( !strcmp( psz_elname, "Genre" ) )
178                     p = &psz_genre;
179                 else if( !strcmp( psz_elname, "Nowplaying" ) )
180                     p = &psz_now;
181                 else if( !strcmp( psz_elname, "Listeners" ) )
182                     p = &psz_listeners;
183                 else if( !strcmp( psz_elname, "Bitrate" ) )
184                     p = &psz_bitrate;
185                 else
186                 {
187                     msg_Warn( p_demux, "unexpected text in element <%s>",
188                               psz_elname );
189                     break;
190                 }
191                 free( *p );
192                 *p = strdup( node );
193                 break;
194             }
195
196             // End element
197             case XML_READER_ENDELEM:
198             {
199                 // Read the element name
200                 if( !strcmp( node, "entry" ) )
201                 {
202                     p_input = input_item_New( psz_mrl, psz_title );
203                     if( psz_now )
204                         input_item_SetNowPlaying( p_input, psz_now );
205                     if( psz_genre )
206                         input_item_SetGenre( p_input, psz_genre );
207                     if( psz_listeners )
208                         msg_Err( p_demux, "Unsupported meta listeners" );
209                     if( psz_bitrate )
210                         msg_Err( p_demux, "Unsupported meta bitrate" );
211
212                     input_item_node_AppendItem( p_subitems, p_input );
213                     vlc_gc_decref( p_input );
214                     FREENULL( psz_title );
215                     FREENULL( psz_mrl );
216                     FREENULL( psz_genre );
217                     FREENULL( psz_bitrate );
218                     FREENULL( psz_listeners );
219                     FREENULL( psz_now );
220                 }
221                 FREENULL( psz_elname );
222                 break;
223             }
224         }
225     }
226
227     if( i_ret < 0 )
228     {
229         msg_Warn( p_demux, "error while parsing data" );
230         i_ret = 0; /* Needed for correct operation of go back */
231     }
232
233 end:
234     free( psz_elname );
235
236     if( p_subitems )
237         input_item_node_PostAndDelete( p_subitems );
238
239     vlc_gc_decref( p_current_input );
240     if( p_xml_reader )
241         xml_ReaderDelete( p_xml_reader );
242     return i_ret;
243 }
244
245 static int Control( demux_t *p_demux, int i_query, va_list args )
246 {
247     VLC_UNUSED(p_demux); VLC_UNUSED(i_query); VLC_UNUSED(args);
248     return VLC_EGENERIC;
249 }
250
251 static bool IsWhitespace( const char *psz_string )
252 {
253     psz_string += strspn( psz_string, " \t\r\n" );
254     return !*psz_string;
255 }