📄 itml.c
字号:
/******************************************************************************* * itml.c : iTunes Music Library import functions ******************************************************************************* * Copyright (C) 2007 the VideoLAN team * $Id$ * * Authors: Yoann Peronneau <yoann@videolan.org> * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA. *******************************************************************************//** * \file modules/demux/playlist/itml.c * \brief iTunes Music Library import functions */#ifdef HAVE_CONFIG_H# include "config.h"#endif#include <vlc_common.h>#include <vlc_demux.h>#include "playlist.h"#include "vlc_xml.h"#include "vlc_strings.h"#include "vlc_url.h"#include "itml.h"struct demux_sys_t{ int i_ntracks;};static int Control( demux_t *, int, va_list );static int Demux( demux_t * );/** * \brief iTML submodule initialization function */int Import_iTML( vlc_object_t *p_this ){ DEMUX_BY_EXTENSION_OR_FORCED_MSG( ".xml", "itml", "using iTunes Media Library reader" ); return VLC_SUCCESS;}void Close_iTML( vlc_object_t *p_this ){ demux_t *p_demux = (demux_t *)p_this; free( p_demux->p_sys );}/** * \brief demuxer function for iTML parsing */int Demux( demux_t *p_demux ){ int i_ret = VLC_SUCCESS; xml_t *p_xml = NULL; xml_reader_t *p_xml_reader = NULL; char *psz_name = NULL; INIT_PLAYLIST_STUFF; p_demux->p_sys->i_ntracks = 0; /* create new xml parser from stream */ p_xml = xml_Create( p_demux ); if( !p_xml ) i_ret = VLC_ENOMOD; else { p_xml_reader = xml_ReaderCreate( p_xml, p_demux->s ); if( !p_xml_reader ) i_ret = VLC_EGENERIC; } /* locating the root node */ if( i_ret == VLC_SUCCESS ) { do { if( xml_ReaderRead( p_xml_reader ) != 1 ) { msg_Err( p_demux, "can't read xml stream" ); i_ret = VLC_EGENERIC; } } while( i_ret == VLC_SUCCESS && xml_ReaderNodeType( p_xml_reader ) != XML_READER_STARTELEM ); } /* checking root node name */ if( i_ret == VLC_SUCCESS ) { psz_name = xml_ReaderName( p_xml_reader ); if( !psz_name || strcmp( psz_name, "plist" ) ) { msg_Err( p_demux, "invalid root node name: %s", psz_name ); i_ret = VLC_EGENERIC; } FREE_NAME(); } if( i_ret == VLC_SUCCESS ) { xml_elem_hnd_t pl_elements[] = { {"dict", COMPLEX_CONTENT, {.cmplx = parse_plist_dict} } }; i_ret = parse_plist_node( p_demux, p_current_input, NULL, p_xml_reader, "plist", pl_elements ); HANDLE_PLAY_AND_RELEASE; } if( p_xml_reader ) xml_ReaderDelete( p_xml, p_xml_reader ); if( p_xml ) xml_Delete( p_xml ); return 0; /* Needed for correct operation of go back */}/** \brief dummy function for demux callback interface */static int Control( demux_t *p_demux, int i_query, va_list args ){ VLC_UNUSED(p_demux); VLC_UNUSED(i_query); VLC_UNUSED(args); return VLC_EGENERIC;}/** * \brief parse the root node of the playlist */static bool parse_plist_node COMPLEX_INTERFACE{ VLC_UNUSED(p_track); VLC_UNUSED(psz_element); char *psz_name = NULL; char *psz_value = NULL; bool b_version_found = false; /* read all playlist attributes */ while( xml_ReaderNextAttr( p_xml_reader ) == VLC_SUCCESS ) { psz_name = xml_ReaderName( p_xml_reader ); psz_value = xml_ReaderValue( p_xml_reader ); if( !psz_name || !psz_value ) { msg_Err( p_demux, "invalid xml stream @ <plist>" ); FREE_ATT(); return false; } /* attribute: version */ if( !strcmp( psz_name, "version" ) ) { b_version_found = true; if( strcmp( psz_value, "1.0" ) ) msg_Warn( p_demux, "unsupported iTunes Media Library version" ); } /* unknown attribute */ else msg_Warn( p_demux, "invalid <plist> attribute:\"%s\"", psz_name); FREE_ATT(); } /* attribute version is mandatory !!! */ if( !b_version_found ) msg_Warn( p_demux, "<plist> requires \"version\" attribute" ); return parse_dict( p_demux, p_input_item, NULL, p_xml_reader, "plist", p_handlers );}/** * \brief parse a <dict> * \param COMPLEX_INTERFACE */static bool parse_dict COMPLEX_INTERFACE{ int i_node; char *psz_name = NULL; char *psz_value = NULL; char *psz_key = NULL; xml_elem_hnd_t *p_handler = NULL; while( xml_ReaderRead( p_xml_reader ) == 1 ) { i_node = xml_ReaderNodeType( p_xml_reader ); switch( i_node ) { case XML_READER_NONE: break; case XML_READER_STARTELEM: /* element start tag */ psz_name = xml_ReaderName( p_xml_reader ); if( !psz_name || !*psz_name ) { msg_Err( p_demux, "invalid xml stream" ); FREE_ATT_KEY(); return false; } /* choose handler */ for( p_handler = p_handlers; p_handler->name && strcmp( psz_name, p_handler->name ); p_handler++ ); if( !p_handler->name ) { msg_Err( p_demux, "unexpected element <%s>", psz_name ); FREE_ATT_KEY(); return false; } FREE_NAME(); /* complex content is parsed in a separate function */ if( p_handler->type == COMPLEX_CONTENT ) { if( p_handler->pf_handler.cmplx( p_demux, p_input_item, NULL, p_xml_reader, p_handler->name, NULL ) ) { p_handler = NULL; FREE_ATT_KEY(); } else { FREE_ATT_KEY(); return false; } } break; case XML_READER_TEXT: /* simple element content */ FREE_ATT(); psz_value = xml_ReaderValue( p_xml_reader ); if( !psz_value ) { msg_Err( p_demux, "invalid xml stream" ); FREE_ATT_KEY(); return false; } break; case XML_READER_ENDELEM: /* element end tag */ psz_name = xml_ReaderName( p_xml_reader ); if( !psz_name ) {
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -