AT89C2051 + SD卡 + 3310LCD = 音乐播放器
扫描二维码
随时随地手机看文章
这个小玩意,采用 ATMEL 的传统51MCU作主控制芯片,加上SD卡和显示屏,就可以作简单的音乐播放器了,虽然音质不怎么样,不过作为DIY还是蛮有乐趣,希望大家喜欢。
没有采用FAT文件系统,只是按扇区读取SD卡,由于2051资源有限,改为4051有望可以操作FAT,但目前程序还在不断完善中。
128byte怎样读取512byte的扇区数据?可以采用边读边播放的方式,就能解决。音乐文件是32KHz取样率的WAV文件,所以和HIFI就沾不上边了。
程序是用C来编写,以方便交流,资料整理中,完善后再上传。
这是未经整理的程序,有点乱,凑合着看,有时间再进一步改进。
SD部分是修改于本坛的一个贴子
----------------------------------------------------------
添加部分注释,提高可读性
#include#include #include #include"LCD_3310.H"#defineucharunsignedchar#defineuintunsignedint#defineulongunsignedlong/************定义管脚*************/sbitDOUT=P3^0;//SD卡数据输出sbitCLK=P3^1;//SD卡时钟输入sbitDIN=P3^2;//SD卡数据输入sbitCS=P3^3;//SD卡片选使能/************全局变量************/ucharpbuf[64];//数据缓冲区ucharp;//播放缓冲区指针ucharpx;//频谱显示的X坐标codeulongTrack[17]={//0x15000,0x58000SD卡中各声音文件的首址,以后打算把这些数据放在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/*********通用延时函数***********/voiddelay(uinti){while(i--);}/********SD写入一个字节**********/voidspi_write(ucharx){//不采用循环结构是为了提高处理速度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慢速写入一个字节********/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读入一字节***********/ucharspi_read(void){//利用51串口的同步移位功能,以达了最高的读度2MHzCLKRI=0;while(RI==0);returnSBUF;}/********SD慢速读入一字节**********/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;}/********发送一组SD命令************/ucharwrite_cmd(uchardata*pcmd){uchartemp,time=0,i;for(i=0;i<6;i++)//一条命令都是6个字节,形参用指针,{//指向6个字节命令,spi_write(pcmd);}do//看看写进去没有,通过so管脚{temp=spi_read();time++;}//一直到读到的不是0xff或超时,退出去while(temp==0xff&&time<100);returntemp;}/******慢速发送一组SD命令**********/ucharwrite_cmd_low_speed(uchar*pcmd){uchartemp,time=0,i;for(i=0;i<6;i++)//一条命令都是6个字节,形参用指针,{//指向6个字节命令,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个时钟周期,{//现在是,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;}/*******读取一扇区的点阵图像*********/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优先,所以结果高低位倒置for(j=0;j<504;j++)//3310的分辨率为84*48,总计用504字节{LCD3310_write_dat(spi_read());}for(x=0;x<10;x++)spi_read();//略过8字节数据和2字节CRCspi_write(0xff);CS=1;return0;}/*******读取一扇区的声音数据*********/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];//随便挑一个数据显示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优先,所以结果高低位倒置while(1)//读取512字节数据{RI=0;_nop_();pbuf[j++&63]=SBUF;//为求快速,不用函数调用RI=0;_nop_();pbuf[j++&63]=SBUF;//直接启动串口移入RI=0;_nop_();pbuf[j++&63]=SBUF;//连续读四字节RI=0;_nop_();pbuf[j++&63]=SBUF;if(j>=512)break;while((((uchar)j-p)&63)>55);//检测播放进度,}//如果缓冲区接近溢出,先暂停等待spi_read();//略过crcspi_read();//略过crcspi_write(0xff);//SD时序要求补8个脉冲CS=1;return0;}/****************************主程序*******************************/intmain(void){ucharkey,n,Count,Min,Sec;ulongaddr;//SD的扇区地址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);//填充播放缓冲区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)//循环播放所有曲目{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