2 UPnP媒体播放器设计的基本思想
2.1 功能描述
UPnP网络媒体播放器可以提供从网络中获取的各种娱乐信息,它允许控制节点对其进行控制。此外,根据所支持协议的不同,播放器也可以提供对数据流进行控制的功能。一个标准的UPnP网络媒体播放器包含播放控制服务、连接管理服务和媒体传输服务。任何媒体播放器都至少提供两种服务:播放控制服务和连接管理服务,媒体传输服务是可选的,它取决于设备所支持的传输协议。这里设计使用的协议是HTTP GET,能提供媒体传输服务,因而可以在数据传输过程中控制数据流。
2.2 过程描述
媒体播放器可由控制节点在局域网内对媒体服务器所提供的媒体进行播放控制。媒体服务器、播放器、控制节点3者的互动过程如下:控制节点使用SSDP协议在局域网内发现一个或多个媒体服务器和媒体播放器,首先定位媒体服务器上的资源,并需明确在服务器和播放器之间传输数据所需的协议和它们都支持的数据格式。这些传输参数都被确定后,控制节点就可以对传输的内容进行控制,如播放、暂停、停止等。真正的数据传输是在服务器和媒体播放器之间直接进行的,并且独立于控制节点,因此不包含在UPnP内。即内容的传输是使用UPnP以外的协议。之后,控制节点使用媒体服务器所提供的内容目录服务来获取该服务器所支持的协议和数据类型,使用媒体播放器的连接管理服务来获取相应的信息,比较后选定双方都支持的传输协议和数据类型。在本设计中,使用的传输协议是HTTP GET,所支持的数据格式为MP3。最后,控制节点使用媒体播放器提供的媒体传输服务来控制数据流。
3 UPnP媒体播放器的实现方法
设计使用Intel公司的开源UPnP开发工具Device-Builder,在Microsoft.NET Framwork下开发。DeviceBuilder生成的UPnP协议栈由MiniServer模块、HTTP模块、线程库模块、XML解析模块及协议栈编程接口等模块组成,负责提供基本的UPnP功能,具体实现流程如下。
3.1 生成UPnP框架
使用DeviceBuilder生成相应平台上的代码,由于是在Microsoft.NET Framwork下开发,所以选择的Target Platform应为:Windows 98,NT,XP。具体功能有:寻址、发现、描述、控制、发布事件。各功能组合在一起,为媒体播放器提供UPnP能力,但Intel开发包生成的仅是一个框架,还要为其添加解码、控制功能及对播放列表的识别等。
3.2 添加媒体解码库
下载一个开源的播放器,将其改造成媒体解码库。即将播放器的功能抽象成函数以供外界调用,主要函数如下:
int decodestart(char*pBuffer):调用该函数开始播放,相当于原来播放器的play功能。pBuffer是一个公共缓冲区,存放已下载的媒体数据。
extem void Read(int*position):负责向公共缓冲区中装入已下载的媒体数据,position指针则对应于公共缓冲区中应装入的位置。由于一般下载的速度远大于解码速度,所以应注意装入新数据时不要将尚未解码的数据覆盖掉。
extern void Decode(int*Dposition):当外界调用该函数时开始解码,Dposition则指示对应于公共缓冲区的解码位置。
此外还有pause,resume,stop等函数,均对应于原播放器的相应功能。将已实现的媒体解码库添加到生成的UPnP框架,再在UPnP框架中相应的位置调用库文件中的函数以实现解码、播放控制。
3.3 播放控制功能实现
添加媒体解码库后。还需添加:数据流实时控制能力、根据播放器的状态发布消息。详细过程如下:
(1)设置传输地址 当控制节点从媒体服务器选定一首歌时,则触发设置传输地址(由媒体传输服务提供)这一动作。这时,如果播放器正在播放,首先要停止播放器的解码动作,为新歌曲的播放做准备。接下来,在播放器获得所选定歌曲的地址后(由控制节点提供),直接从媒体服务器中下载这首歌曲,在此项目中,播放器采用HTTP GET下载。下载前,还需对该URL进行判断,查看下载的是否为播放列表,如果是播放列表,就不能将其直接传给播放器,而要继续从播放列表提供的地址中得到第1首歌的URL,再从媒体服务器中下载。最后,调用函数SetLastChange发布自己的状态信息。
(2)数据流实时控制在下载的大小达到一定长度时,触发传输控制服务中Play这一动作,这时,主线程创建一个解码线程,该线程调用媒体解码库中的decodestart(char*pBuffer)函数,对存放在公用缓冲区中的数据进行解码并播放,进行同步控制,以免当网络出现拥塞或系统内存不足时,缓冲区中的播放指针超过下载指针而导致意外。如果选定的是含有多首歌曲的播放列表,则过程类似,只不过在一首歌曲播放完时会自动从媒体服务器端下载另一首歌曲,一边下载,一边播放。当用户点击next,previous时,首先检查之前保存的URL是否为播放列表,若不是,则输出错误信息;若是,则停止播放器的解码动作,撤销当前解码线程,为新歌的播放做准备。由播放列表得到下一首歌的地址和端口号,开始下载数据,并创建新的解码线程,最后,发布状态信息。