FLAC无损音频播放
象棋小子 1048272975
无损音频不存在音频损失的问题,可以获取到原始PCM采样数据,代表数字音频最高的保真水准。随着储存设备容量的增大,网络传输带宽的提升,无损音频越来越受到人们的追捧。
1. FLAC概述常用的无损音频格式有WAV、FLAC、APE等等。WAV格式直接保存原始的PCM数据,造成容量过大,不方便使用。FLAC和APE为无损音频压缩格式,通过压缩算法重新对PCM数据进行编码,减小容量,同时可以通过可逆的算法解压还原成PCM数据。在众多的无损音频压缩格式中,FLAC被认为是最具潜力的一种无损音频压缩格式。
FLAC(FreeLossless Audio Codec)是一套自由的音频压缩编解码器,不同于其它有损音频压缩格式,FLAC格式不会破坏任何原有的音频资讯。这种压缩类似于ZIP的方式,但FLAC是专门针对PCM音频特点设计的压缩格式,其压缩率要大于ZIP方式。FLAC是一种非专有的,不受专利影响,开放源码,并且完全免费的无损音频压缩格式,它的编码算法相当成熟,已经通过了严格的测试,被很多软件以及硬件音频产品所支持,如支持大多数的操作系统,包括Windows、Unix类系统、Mac等等,应用在移动多媒体播放器、汽车音响、家用音响等等设备。
2、FLAC库移植FLAC编解码复杂度比较低,对计算要求不高,在普通的硬件平台就可以轻松实现实时解码播放,因此可以在LPC5411x开发平台上实现FLAC音频的播放。FLAC有专门的项目组维护,可以在https://xiph.org/flac下载完整的FLAC编解码源码,对于C开发环境,其对应的库为libFLAC。libFLAC包含了操作系统相关的数据结构,并且对于cortex-m平台来说,功能过于臃肿,需要对该库做较大的改动才能完成移植,并不是最适合的。
此处采用rockbox嵌入式项目中的FLAC解码库,减少移植的工作量。rockbox是一个免费的数字音频播放器框架,支持众多的音频编解码器。其采用的FLAC解码库移植于libffmpeg,与FLAC解码相关的源码有:
bitstream.c/bitstream.h,获取位流源码实现。
decoder.c/decoder.h,FLAC解码实现。
arm.S/arm.h,针对cortex-m,MDK下修改的LPC解码arm汇编实现,非必须文件,用于优化Level 8的解码速度。在decoder.c中取消定义CPU_ARM宏则取消arm汇编的优化实现。
golomb.h和tables.c为编码头文件和常数表实现。
把该库源码加入LPC5411x工程编译即可。
3、FLAC播放FLAC音频的播放涉及到音频驱动、SD卡读写文件的实现,可以参考前面的章节。播放实现主要流程如下:
a. 用Flac_ParceMetadata()函数打开FLAC音频文件并解析FLAC头,获取采样位数、采样频率、声道数等等音频格式。
static int Flac_ParceMetadata(constTCHAR filePath[], FLACContext* context)
{
UINT s1 = 0;
FIL FLACfile;
int metaDataFlag = 1;
char metaDataChunk[128];
unsigned long metaDataBlockLength = 0;
char* tagContents;
if(f_open(&FLACfile, filePath,FA_READ) != FR_OK) {
PRINTF("Couldnot open: %s\r\n", filePath);
return1;
}
f_read(&FLACfile, metaDataChunk, 4,&s1);
if(s1 != 4) {
PRINTF("Readfailure\r\n");
f_close(&FLACfile);
return1;
}
if(memcmp(metaDataChunk,"fLaC", 4) != 0) {
PRINTF("Nota FLAC file\r\n");
f_close(&FLACfile);
return1;
}
// Now we are at the stream block
// Each block has metadata header of 4bytes
do {
f_read(&FLACfile, metaDataChunk, 4,&s1);
if(s1 != 4) {
PRINTF("Readfailure\r\n");
f_close(&FLACfile);
return1;
}
//Check if last chunk
if(metaDataChunk[0] & 0x80)metaDataFlag = 0;
metaDataBlockLength = (metaDataChunk[1]<< 16) | (metaDataChunk[2] << 8)
| metaDataChunk[3];
//STREAMINFO block
if((metaDataChunk[0] & 0x7F) == 0) {
if(metaDataBlockLength> 128) {
PRINTF("Metadatabuffer too small\r\n");
f_close(&FLACfile);
return1;
}
f_read(&FLACfile,metaDataChunk, metaDataBlockLength, &s1);
if(s1!= metaDataBlockLength) {
PRINTF("Readfailure\r\n");
f_close(&FLACfile);
return1;
}
/*
<bits>Field in STEAMINFO
<16>min block size (samples)
<16>max block size (samples)
<24>min frams size (bytes)
<24>max frams size (bytes)
<20>Sample rate (Hz)
<3>(number of channels)-1
<5>(bits per sample)-1.
<36>Total samples in stream.
<128>MD5 signature of the unencoded audio data.
*/
context->min_blocksize= (metaDataChunk[0] << 8) | metaDataChunk[1];
context->max_blocksize= (metaDataChunk[2] << 8) | metaDataChunk[3];
context->min_framesize= (metaDataChunk[4] << 16) |
(metaDataChunk[5] << 8) | metaDataChunk[6];
context->max_framesize= (metaDataChunk[7] << 16) |
(metaDataChunk[8] << 8)| metaDataChunk[9];
context->samplerate= (metaDataChunk[10] << 12) |
(metaDataChunk[11] << 4) | ((metaDataChunk[12] & 0xf0)>> 4);
context->channels= ((metaDataChunk[12] & 0x0e) >> 1) + 1;
context->bps= (((metaDataChunk[12] & 0x01) << 4) |
((metaDataChunk[13] &0xf0)>>4) ) + 1;
//Thisfield in FLAC context is limited to 32-bits
context->totalsamples= (metaDataChunk[14] << 24) | (metaDataChunk[15] << 16) |
(metaDataChunk[16] <<8) | metaDataChunk[17];
} else {
if(f_lseek(&FLACfile,FLACfile.fptr + metaDataBlockLength) != FR_OK) {
PRINTF("FileSeek Failed\r\n");
f_close(&FLACfile);
return1;
}
}
} while(metaDataFlag);
// track length in ms
context->length =(context->totalsamples / context->samplerate) * 1000;
// file size in bytes
context->filesize =f_size(&FLACfile);
// current offset is end of metadata inbytes
context->metadatalength =FLACfile.fptr;
// bitrate of file
context->bitrate =((context->filesize - context->metadatalength) * 8) / context->length;
f_close(&FLACfile);
return 0;
}
b. 根据解析的音频格式,对I2S音频驱动初始化。
PRINTF("Playing %s\r\n",filePath);
PRINTF("Mode: %s\r\n",context.channels==1?"Mono":"Stereo");
PRINTF("Samplerate: %d Hz\r\n",context.samplerate);
PRINTF("SampleBits: %dbit\r\n", context.bps);
PRINTF("Samples: %d\r\n",context.totalsamples);
I2S_SetSamplerate(context.samplerate);
I2S_TxStart();
c. 调用flac_decode_frame()从fileChunk缓存区读取一块的码流数据进行解码,解码出来的左声道数据放在decodedSamplesLeft缓存区,右声道数据放在decodedSamplesRight缓存区。
if (flac_decode_frame(&context,decodedSamplesLeft, decodedSamplesRight,
fileChunk,bytesLeft, yield) < 0) {
PRINTF("FLACDecode Failed\r\n");
break;
}
d. 解码的左右声道数据填充到音频输出缓存,进行播放。解码一块产生context.blocksize个音频数据,一个一个填充到音频输出缓存,如果输出缓存满,则等待播放完一帧后,继续填充。
i = 0;
while (i < context.blocksize) {
//LeftChannel
samplePair[0]= (uint16_t) (decodedSamplesLeft>>sampleShift);
if(context.channels==2) {
//RightChannel
samplePair[1]= (uint16_t) (decodedSamplesRight>>sampleShift);
}else {
//RepeatLeft channel if mono
samplePair[1]= (uint16_t) (decodedSamplesLeft>>sampleShift);
}
while(WriteIndex == I2SState.TxReadIndex) {
}
//Samplepair is 4 bytes, 16-bit mode
if(WriteIndex != I2SState.TxReadIndex) {
I2SState.TxBuffer[I2SState.TxWriteIndex][Index]=
(samplePair[0]&0xffff) |(samplePair[1]<<16);
Index++;
if(Index >= AUDIO_FRAME_SIZE) {
Index= 0;
I2SState.TxWriteIndex= WriteIndex;
if(WriteIndex >= AUDIO_NUM_BUFFERS-1) {
WriteIndex= 0;
}else {
WriteIndex++;
}
}
}
i++;
}
e. 从SD卡读取下一块码流数据到fileChunk缓存区,跳到步骤c,如此循环解码、填充音频输出流、读取码流数据这些过程,直至文件结束,解码完毕。
//calculate the number of valid bytesleft in the fileChunk buffer
bytesUsed = context.gb.index/8;
bytesLeft -= bytesUsed;
//shift the unused stuff to the front ofthe fileChunk buffer
memmove(fileChunk,&fileChunk[bytesUsed], bytesLeft);
//Refill the fileChunk buffer
f_read(&FLACfile,&fileChunk[bytesLeft], MAX_FRAMESIZE - bytesLeft, &s1);
//add however many were read
bytesLeft += s1;
播放<<天空之城>>主题曲sky city.flac如下:
4. 附录
MDK工程,包含SD卡文件读写代码,I2S播放驱动,FLAC无损音频文件解码播放的实现,FLAC音频文件sky city.flac。
https://pan.baidu.com/s/1boEFZ6R
- 妤傛ḿ楠囩亸鍕暥瀹搞儳鈻肩敮鍫濆悋閹存劕鐓跨拋顓熸殌缁嬪顨滅憗锟�
閸忋劍鏌熸担宥咁劅娑旂姴鐨犳0鎴滅瑩娑撴氨鐓$拠鍡礉閹绘劕宕岄惍鏂垮絺瀹搞儰缍旈懗钘夊閿涘苯濮幃銊ユ彥闁喐鍨氶梹澶歌礋娴兼ḿ顫呴惃鍕殸妫版垵浼愮粙瀣瑎...
- 娑擃厾楠囩亸鍕暥瀹搞儳鈻肩敮鍫濆悋閹存劕鐓跨拋顓熸殌缁嬪顨滅憗锟�
缁箖鈧拷30婢舵岸妫亸鍕暥閸╃顔勭拠鍓р柤閿涘奔绗撶€硅埖宸跨拠鎾呯礉閸斺晛顒熼崨妯烘彥闁喕鎻崚棰佺娑擃亜鎮庨弽鐓庣殸妫版垵浼愮粙瀣瑎閻ㄥ嫯顩﹀Ч锟�...
- Agilent ADS 閺佹瑥顒熼崺纭咁唲鐠囧墽鈻兼總妤勵棅
娑撴挸顔嶉幒鍫n嚦閿涘苯鍙忛棃銏n唹鐟欘枃DS閸氬嫮顫掗崝鐔诲厴閸滃苯浼愮粙瀣安閻㈩煉绱遍崝鈺傚亶閻€劍娓堕惌顓犳畱閺冨爼妫跨€涳缚绱癆DS...
- HFSS鐎涳缚绡勯崺纭咁唲鐠囧墽鈻兼總妤勵棅
鐠у嫭绻佹稉鎾愁啀閹哄牐顕抽敍灞藉弿闂堛垼顔夐幒鍦欶SS閻ㄥ嫬濮涢懗钘夋嫲鎼存梻鏁ら敍灞藉簻閸斺晜鍋嶉崗銊╂桨缁崵绮洪崷鏉款劅娑旂姵甯夐幓顡嶧SS...
- CST瀵邦喗灏濆銉ょ稊鐎广倕鐓跨拋顓熸殌缁嬪顨滅憗锟�
閺夊孩妲戝ú瀣╁瘜鐠佽绱濋崗銊╂桨鐠佸弶宸緾ST閸氬嫰銆嶉崝鐔诲厴閸滃苯浼愮粙瀣安閻㈩煉绱濋崝鈺傚亶韫囶偊鈧喕鍤滅€涳附甯夐幓顡塖T鐠佹崘顓告惔鏃傛暏...
- 鐏忓嫰顣堕崺铏诡攨閸╃顔勭拠鍓р柤
娑撳洣绗€妤傛ɑ銈奸獮鍐叉勾鐠у嚖绱濇潻娆庣昂鐠囧墽鈻兼稉杞扮稑閸︺劌鐨犳0鎴炲Η閺堫垶顣崺鐔枫亣鐏炴洘瀚甸懘姘剧礉閹垫挷绗呴崸姘杽閻ㄥ嫪绗撴稉姘唨绾偓...
- 瀵邦喗灏濈亸鍕暥濞村鍣洪幙宥勭稊閸╃顔勭拠鍓р柤閸氬牓娉�
鐠愵厺鎷遍崥鍫ユ肠閺囨潙鐤勯幆鐙呯礉缂冩垵鍨庨妴渚€顣剁拫鍙樺崕閵嗕胶銇氬▔銏犳珤閵嗕椒淇婇崣閿嬬爱閿涘本鍨滅憰浣圭壉閺嶉绨块柅锟�...