AT89C2051 + SD卡 + 3310LCD = 音樂播放器
這個小玩意,采用 ATMEL 的傳統(tǒng)51MCU作主控制芯片,加上SD卡和顯示屏,就可以作簡單的音樂播放器了,雖然音質不怎么樣,不過作為DIY還是蠻有樂趣,希望大家喜歡。
沒有采用FAT文件系統(tǒng),只是按扇區(qū)讀取SD卡,由于2051資源有限,改為4051有望可以操作FAT,但目前程序還在不斷完善中。
128byte怎樣讀取512byte的扇區(qū)數(shù)據(jù)?可以采用邊讀邊播放的方式,就能解決。音樂文件是32KHz取樣率的WAV文件,所以和HIFI就沾不上邊了。
程序是用C來編寫,以方便交流,資料整理中,完善后再上傳。
這是未經(jīng)整理的程序,有點亂,湊合著看,有時間再進一步改進。
SD部分是修改于本壇的一個貼子
----------------------------------------------------------
添加部分注釋,提高可讀性
#include#include #include #include"LCD_3310.H"#defineucharunsignedchar#defineuintunsignedint#defineulongunsignedlong/************定義管腳*************/sbitDOUT=P3^0;//SD卡數(shù)據(jù)輸出sbitCLK=P3^1;//SD卡時鐘輸入sbitDIN=P3^2;//SD卡數(shù)據(jù)輸入sbitCS=P3^3;//SD卡片選使能/************全局變量************/ucharpbuf[64];//數(shù)據(jù)緩沖區(qū)ucharp;//播放緩沖區(qū)指針ucharpx;//頻譜顯示的X坐標codeulongTrack[17]={//0x15000,0x58000SD卡中各聲音文件的首址,以后打算把這些數(shù)據(jù)放在SD卡的特定配置文件中再讀入。0xd7800-0x8a00,0x76b800-0x8a00,0xedc000-0x8a00,0x1752800-0x8a00,0x1F08000-0x8a00,0x2569800-0x8a00,0x2EDB800-0x8a00,0x3480000-0x8a00,0x3BFA800-0x8a00,0x41EB000-0x8a00,0x48EF000-0x8a00,0x508A000-0x8a00,0x59AE800-0x8a00,0x60AF000-0x8a00,0x6878000-0x8a00,0x6DBE000-0x8a00,0x7525800-0x8a00,};/*******SD訪問錯誤碼的定義*******/#defineINIT_CMD0_ERROR0X01#defineINIT_CMD1_ERROR0X02#defineREAD_BLOCK_ERROR0X03#defineWRITE_BLOCK_ERROR0X04/*********通用延時函數(shù)***********/voiddelay(uinti){while(i--);}/********SD寫入一個字節(jié)**********/voidspi_write(ucharx){//不采用循環(huán)結構是為了提高處理速度DIN=x&0x80;CLK=0;CLK=1;DIN=x&0x40;CLK=0;CLK=1;DIN=x&0x20;CLK=0;CLK=1;DIN=x&0x10;CLK=0;CLK=1;DIN=x&0x08;CLK=0;CLK=1;DIN=x&0x04;CLK=0;CLK=1;DIN=x&0x02;CLK=0;CLK=1;DIN=x&0x01;CLK=0;CLK=1;}/*******SD慢速寫入一個字節(jié)********/voidspi_write_low_speed(ucharx){uchari;for(i=8;i;--i){DIN=x&0x80;x<<=1;CLK=0;delay(1);CLK=1;delay(1);}}/***********SD讀入一字節(jié)***********/ucharspi_read(void){//利用51串口的同步移位功能,以達了最高的讀度2MHzCLKRI=0;while(RI==0);returnSBUF;}/********SD慢速讀入一字節(jié)**********/ucharspi_read_low_speed(void){uchartemp,i;for(i=8;i;--i){CLK=0;delay(1);temp<<=1;if(DOUT)temp++;CLK=1;delay(1);}returntemp;}/********發(fā)送一組SD命令************/ucharwrite_cmd(uchardata*pcmd){uchartemp,time=0,i;for(i=0;i<6;i++)//一條命令都是6個字節(jié),形參用指針,{//指向6個字節(jié)命令,spi_write(pcmd);}do//看看寫進去沒有,通過so管腳{temp=spi_read();time++;}//一直到讀到的不是0xff或超時,退出去while(temp==0xff&&time<100);returntemp;}/******慢速發(fā)送一組SD命令**********/ucharwrite_cmd_low_speed(uchar*pcmd){uchartemp,time=0,i;for(i=0;i<6;i++)//一條命令都是6個字節(jié),形參用指針,{//指向6個字節(jié)命令,spi_write_low_speed(pcmd);}do//看看寫進去沒有,通過so管腳{temp=spi_read_low_speed();time++;}//一直到讀到的不是0xff或超時,退出去while(temp==0xff&&time<100);returntemp;}/*********SD卡激活,復位*********/ucharsd_reset(void){uchartime,temp,i;ucharpcmd[6]={0x40,0x00,0x00,0x00,0x00,0x95};CS=1;for(i=0;i<0x0f;i++)//復位時,至少要72個時鐘周期,{//現(xiàn)在是,15*8=120個clkspi_write_low_speed(0xff);}CS=0;time=0;do{temp=write_cmd_low_speed(pcmd);time++;if(time>100)returnINIT_CMD0_ERROR;}while(temp!=0x01);//校驗碼是0x01,表示寫入成功CS=1;spi_write_low_speed(0xff);//時序上要求補8個clkreturn0;//返回0,寫入成功}/************SD卡初始化************/ucharsd_init(void){uchartime,temp;ucharpcmd[6]={0x41,0x00,0x00,0x00,0x00,0xff};CS=0;time=0;do{temp=write_cmd_low_speed(pcmd);time++;if(time>100)returnINIT_CMD1_ERROR;}while(temp!=0x00);CS=1;spi_write_low_speed(0xff);return0;}/*******讀取一扇區(qū)的點陣圖像*********/ucharsd_read_bmp(uchardata*ad){uchartemp,time,x,pcmd[6];uintj=0;pcmd[0]=0x51;pcmd[1]=*ad;pcmd[2]=*(++ad);pcmd[3]=*(++ad);pcmd[4]=0;pcmd[5]=0xff;CS=0;time=0;do{temp=write_cmd(pcmd);if(++time>100){CS=1;returnREAD_BLOCK_ERROR;}}while(temp!=0);//等待SD卡回應while(spi_read()!=0x7f);//0xfe,51的串口移位是LSB優(yōu)先,所以結果高低位倒置for(j=0;j<504;j++)//3310的分辨率為84*48,總計用504字節(jié){LCD3310_write_dat(spi_read());}for(x=0;x<10;x++)spi_read();//略過8字節(jié)數(shù)據(jù)和2字節(jié)CRCspi_write(0xff);CS=1;return0;}/*******讀取一扇區(qū)的聲音數(shù)據(jù)*********/ucharsd_read_sector(uchardata*ad){uchartemp,time,pcmd[6];uintj=0;pcmd[0]=0x51;pcmd[1]=*ad;pcmd[2]=*(++ad);pcmd[3]=*(++ad);pcmd[4]=0;pcmd[5]=0xff;CS=0;time=0;do{temp=write_cmd(pcmd);if(++time>100){CS=1;returnREAD_BLOCK_ERROR;}}while(temp!=0);//等待SD回應的時間有點長,所以在這里插入顯示模擬的頻譜圖temp=pbuf[16];//隨便挑一個數(shù)據(jù)顯示LCD3310_set_XY(px,5);//設定顯示位置px+=6;if(px>=39)px=0;if(temp&0x80)temp^=0x80;//求得聲音振幅elsetemp=0x80-temp;temp=Level[temp>>4];//不同幅度對應不同的譜線圖案LCD3310_write_dat(temp);LCD3310_write_dat(temp);LCD3310_write_dat(temp);while(spi_read()!=0x7f);//0xfe,51的串口移位是LSB優(yōu)先,所以結果高低位倒置while(1)//讀取512字節(jié)數(shù)據(jù){RI=0;_nop_();pbuf[j++&63]=SBUF;//為求快速,不用函數(shù)調用RI=0;_nop_();pbuf[j++&63]=SBUF;//直接啟動串口移入RI=0;_nop_();pbuf[j++&63]=SBUF;//連續(xù)讀四字節(jié)RI=0;_nop_();pbuf[j++&63]=SBUF;if(j>=512)break;while((((uchar)j-p)&63)>55);//檢測播放進度,}//如果緩沖區(qū)接近溢出,先暫停等待spi_read();//略過crcspi_read();//略過crcspi_write(0xff);//SD時序要求補8個脈沖CS=1;return0;}/****************************主程序*******************************/intmain(void){ucharkey,n,Count,Min,Sec;ulongaddr;//SD的扇區(qū)地址P1=0x80;//DAC輸出中點電壓RI=1;REN=1;TMOD=0x02;TH0=256-62.5;//定時器設定約為32KHz,和WAV文件取樣率對應ET0=1;EA=1;px=0;n=64;dopbuf[--n]=0x80;while(n);//填充播放緩沖區(qū)delay(65535);LCD3310_init();LCD3310_set_XY(0,0);LCD3310_write_cmd(0x22);//設定LCD掃描順序sd_reset();sd_init();addr=0x4f400;sd_read_bmp((uchar)&addr);//顯示歡迎畫面while(D_C==1);while(D_C==0);//等待按鍵delay(65535);//==============mainloop==================while(1)//循環(huán)播放所有曲目{TR0=0;LCD3310_write_cmd(0x22);LCD3310_set_XY(0,0);addr=0x4f600+((uint)n<<9);sd_read_bmp((uchar)&addr);//顯示歌名、歌手LCD3310_write_cmd(0x20);TR0=1;p=0xd0;Min=0;Sec=0;Count=0;for(addr=Track[n];addr