「技术分享」音乐-CD*放播**介绍

音乐(deepin-music)是deepin自研的一款音频*放播**软件,能广泛支持*放播**本地或外部媒介中的音频文件,其中就包括对CDDA(Compact Disc-Digital Audio)标准的支持。CDDA标准主要用于光盘存储数字高保真音乐,深受音乐发烧友的喜爱。

「技术分享」音乐-CD*放播**介绍

本文将站在技术实现的角度,介绍音乐(deepin-music)应用是如何*放播**CD光盘中的美妙音乐的。

「技术分享」音乐-CD*放播**介绍

1. 整体流程

简单的说,音乐(deepin-music)*放播**CD流程整体分为三个部分:

  • 检测CD光盘;
  • 解析CD数据;
  • *放播**CD;

「技术分享」音乐-CD*放播**介绍

图一 deepin-music CD光盘处理流程

2. 检测CD光盘

2.1 识别光驱设备

对光驱设备的识别,主要源于对CD光盘状态的检测。依托于 `libudisks2-qt5-dev` 开源库的DDiskManager 类,即可实现对块设备状态监控,具体使用如下:

DDiskManager diskManager;
//获取块设备状态
diskManager.setWatchChanges(true);
//初始化获取当前所有设备localPath
QStringList allDevices = diskManager.blockDevices(QVariantMap());

//根据cdrom关键字,找出光驱设备
foreach (QString tmpstr, allDevices)
{
   if (tmpstr.compare("sr0") == 0) //sr0是cdrom关键字
       return tmpstr; //localPath
}

Linux系统中, "sr0" 主要指光驱设备(s= scsi;r= rom ;数字0代表光驱设备编号)。如需判断当前光驱是否被插拔,则可通过DDiskManager 类以下信号进行监听:

void blockDeviceAdded(const QString &path);void blockDeviceRemoved(const QString &path);

3. 解析CD数据

3.1 解析CD节点

识别到光驱设备后,接下来就是获取CD光盘的节点数据,这里可引用依赖库`libvlc-dev`和`libvlccore-dev`,具体代码实现如下:

// 将tmpstr 转化为"cdda:///dev/sr0"这种形式QString strcda = QString("cdda: ///dev/%1").arg(tmpstr.mid(tmpstr.lastIndexOf("/") + 1, tmpstr.size() - tmpstr.lastIndexOf("/"))) ;// 获取sr0input_item_t *p_input =input_item_NewExt(strcda.toUtf8().data(), "access_demux", 0, ITEM_TYPE_DISC, ITEM_LOCAL); //打开sr0,读取流libvlc_instance_t *_vlcInstance = vlc_new(0, nullptr); //创建libvlc_instance_t对象libvlc_media_player_t *_vlcMediaPlayer = vlc_media_player_new(_vlcInstance); // player_tstream_t * pstream = vlc_stream_NewURL((vlc_object_t *)_vlcMediaPlayer, strcda.toUtf8().data()); //根据sr0,获取流//创建cda节点input_item_node_t * p_items = input_item_node_Create(p_input);//将数据流读到节点中vlc_stream_ReadDir(pstream, p_items);//释放数据流、节点vlc_stream_Delete(pstream); //释放数据流input_item_Release(p_input); //释放输入设备

最终产生的p_items,就包含有CD节点数据。

3.2 检索CD歌曲信息

一般而言,获取了CD节点数据已经足以支持音乐的完整*放播**,但在某些场景下有的用户希望知悉CD歌曲的完整信息(比如:专辑、歌曲名称、歌手),而这部分信息并不在CD光盘里,那么怎么才能获取呢?

「技术分享」音乐-CD*放播**介绍

可以考虑查询远程CD数据库(CDDB),远程CD数据库有gundb、Gracenote和微软CDDB等。这里只介绍gundb使用方法,在连接数据库时需要依赖`libcddb2-dev`库,代码实现如下:

#include <cddb/cddb.h>#include <linux/cdrom.h>//打开/dev/sr0设备文件int fd=open(“/dev/sr0”, O_RDONLY | O_NONBLOCK);//获取CD数据int status = ioctl(fd, CDROM_DISC_STATUS, CDSL_CURRENT);if ((CDS_AUDIO==status || CDS_MIXED==status) && 0==ioctl(fd, CDROMREADTOCHDR, &th)){m_disc = cddb_disc_new(); //生成disccddb_disc_calc_discid(m_disc); //创建discid,后面的查找就是根据它}//创建数据库连接QString host("gnudb.gnudb.org"); //Locationint port = 80;//portcddb_conn_t *connection = cddb_new(); //创建cddb_conn_t 对象cddb_cache_disable(connection);cddb_set_server_name(connection, host.toLatin1().constData()); //设置服务名cddb_set_server_port(connection, port); //设置端口//创建服务cddb_cache_disable(connection);cddb_set_server_name(connection, QString("gnudb.gnudb.org").constData());cddb_set_server_port(connection, 80);//设置http服务cddb_set_http_proxy_server_name(connection, hostName().toLatin1().constData());cddb_set_http_proxy_server_port(connection, port());cddb_http_proxy_enable(connection);cddb_query(connection, m_disc); //查询cddb_read(connection, m_disc); //读取cddb_disc_get_title(m_disc); //读取专辑标题cddb_disc_get_artist(m_disc); //读取专辑作者cddb_track_t *trk=cddb_disc_get_track(m_disc, t); //获取cd中的第t首歌cddb_track_get_title(trk); //获取歌曲标题cddb_track_get_artist(trk); //获取歌曲作者cddb_query_next(connection, m_disc); //查询下一条数据

注:CDDB(光盘数据库)服务器,是可以被 CD翻录器和CD *放播**器等应用程序查询,提供给定CD 的作者、专辑、曲目列表和其他信息。

4. *放播**CD

音频*放播**会沿用3.1章节的相关依赖库,使用`libvlc_instance_t` 和 `libvlc_media_player_t `等接口*放播**CD音频文件。

//在3.1节中,已经初始化得到libvlc_instance_t *_vlcInstance、//libvlc_media_player_t *_vlcMediaPlayer//location路径“cdda:///dev/sr0”libvlc_media_t* _vlcMedia = libvlc_media_new_location(_vlcInstance, path.toUtf8().data());//设置与libvlc_media_player_t关联的媒体描述libvlc_media_player_set_media(_vlcMediaPlayer, _vlcMedia);//track为CD中歌曲的编号,可以自定义指定,cdda-track为CD标识config_PutInt((vlc_object_t *)_vlcMediaPlayer, "cdda-track", track);//*放播**libvlc_media_player_play(_vlcMediaPlayer);

以上便是deepin-music*放播**CD光盘的完整处理流程,希望此篇文章能对感兴趣的朋友起到抛砖引玉的作用。