600字范文,内容丰富有趣,生活中的好帮手!
600字范文 > 毕业设计 单片机音乐播放器设计 - 物联网 嵌入式 Stm32

毕业设计 单片机音乐播放器设计 - 物联网 嵌入式 Stm32

时间:2022-08-09 19:24:58

相关推荐

毕业设计 单片机音乐播放器设计 - 物联网 嵌入式 Stm32

文章目录

0 前言1 简介2 主要器件3 实现效果4 设计原理5 部分核心代码6 最后

0 前言

🔥 这两年开始毕业设计和毕业答辩的要求和难度不断提升,传统的毕设题目缺少创新和亮点,往往达不到毕业答辩的要求,这两年不断有学弟学妹告诉学长自己做的项目系统达不到老师的要求。

为了大家能够顺利以及最少的精力通过毕设,学长分享优质毕业设计项目,今天要分享的是

🚩基于Stm32单片机的音乐播放器设计与实现

🥇学长这里给一个题目综合评分(每项满分5分)

难度系数:3分工作量:3分创新点:3分

1 简介

采用STM32实现的多功能音乐播放器。

2 主要器件

STM32F103RBT6主控芯片VS1003解码芯片(解码MP3)TEA5767立体收音机芯片DS18B20数字温度温度传感器RGB彩灯EEPROM芯片TPA152功率放大芯片LM2576-3.3V电源芯片

3 实现效果

整体实物

系统主界面

音乐播放器功能部分展示

4 设计原理

硬件系统框图

MCU为整个系统的核心,控制着整个系统的运行,让MCU稳定的运行是非常必须的,下图(图2.2)为MCU的原理图,包括一个后备电源UPS1,一个主电源VCC3.3和一个模拟电源,模拟电源通过从VCC3.3加滤波电路得到。MCU外围的必须电路由滤波电容,下载电路(串口1)以及复位开关组成。同时,考虑到系统需要时钟功能,给时钟部分增加了后背电源电路,通过二极管连接到VBAT脚,给实时时钟供电。这里采用了双电源结构,即在电源有外部供电的时候,后备电池不给时钟供电,时钟的电源来自外部,只有当外部电源断开的时候,后备电源才给时钟供电,以保持时钟的计时,这样可以延长后备电池的使用时间。

同时,为了调试方便,下面电路还加了一个多余的按键和LED灯,方便在调试的时候使用。并且,考虑到某些模块对速度的要求,特意对MCU的IO口做了安排,这样虽然增加了布线难度,但是提高了执行速度,还是值得的。对多余IO口的安排,则是全部引出,方便以后扩展其他功能,比如:家电控制等。同时,对于STM32F103RBT6自带的USB接口,也已经引出,日后通过升级,可以实现USB控制的功能。

这里要注意一点:因为PT2314,TEA5767,FM24C16这三个器件都是使用IIC总线控制的,所以,把这三个器件挂在一个IIC总线上,节省了IO口。MCU和DS18B20模块电路图如下:

软件模块化设计

对于底层驱动软件子系统包括如下模块程序:LCD驱动模块、触摸屏驱动模块、SD卡驱动模块、VS1003驱动模块、PT2314驱动模块、FM24C16驱动模块、TEA5767驱动模块、温度传感器驱动模块、彩灯驱动模块、实时时钟驱动模块。对于应用软件子系统包括如下模块程序:JPEG/BMP解码模块、FAT文件系统管理模块、音乐播放模块、图片浏览模块、游戏模块、闹钟模块、时间模块、设置管理模块、电子书模块、收音机模块、彩灯控制模块。

LCD模块驱动程序设计

本系统用到的LCD是八位数据模式,驱动IC型号是FMT0371,该芯片为松下合资厂生产的一个LCD驱动IC。最高支持26万色的TFT LCD,有6位、8位、16位和18位数据模式,可以方便选择。本系统配套的LCD使用的是八位数据模式,65K色。

根据该LCD的DATASHEET,每个像素点的GRAM实际上是一个18bit的数据寄存器。在16bit模式下与写入数据的对应关系如图3.1 所示:

从图中可以看出,RGB的有效位数分别为565,比如写入0XF800则显示纯红色,写入0X07E0则显示纯绿色,写入0X001F 则显示纯蓝色。在处理数据的时候要把像素值先变换为这样的结构,然后再写入LCD。LCD的显示状态都是由LCD的控制命令控制的,通过写入不同的控制命令和数据,就可以实现不同的现实功能和效果。分析DATASHEET得到几个重要的控制命令:

00H:这个命令用来控制内存操作模式,这里我们主要用它来改变LCD的扫描方向。

02H,03H:这两个命令用来分别设置X,Y方向的开始显示的点坐标。

04H,05H:这两个命令用来分别设置X,Y方向的结束显示的点坐标。

0EH,0FH:这两个命令用来写入和读取显存。

LCD驱动部分包括几个关键函数:LCD读写寄存器函数、LCD读写数据函数、LCD初始化函数和LCD画点函数。有了这几个基本函数,其他的画线、画面、甚至画图都比较容易了。LCD与MCU的连线包括D0~D7、CS、RS、RST、WR、RD、BL共14根线。

D0~D7:数据线

CS:LCD的片选线,低电平有效。

RS:LCD的地址/数据控制,高电平表示数据,低电平表示地址。

RST:复位线,低电平有效。

WR:写数据访问控制。

RD:读数据访问控制。

BL:LCD背光,高电平有效。

LCD读写寄存器:对LCD寄存器的操作线设置RS为低,表示写入寄存器,然后拉低片选信号,给BL

送入数据,然后通过一个WR的脉冲,就可以把数据写入到LCD了。最后释放RS,CS,完成此次操作。对LCD寄存器的读操作和写操作差不多,不同之处就是把WR脉冲改为RD脉冲。LCD读写数据:对于LCD数据的读写,和寄存器的读写差不多,只要把RS设置为高,就表示此次操作是对数据的读写,其他同寄存器的读写操作一样。LCD初始化:这部分是在前面两步成功的基础上才能进行的,LCD的初始化涉及到其内部很多寄存器的初始化。比较复杂,由void TFT_Init(void)函数实现,具体初始化过程请参考附件里面的代码。LCD画点:画点的实现,要先设置LCD开始显示和结束显示的范围,通过0X02H~0X04H这四个命令实现。之后写入0X0E命令,开始写入数据,就可以写入像素值(16bit)了,对于画点,我们只要写入一个像素点就可以了,这样就完成了在LCD上画一点。具体见附件里面的void TFT_DrawPoint(u8 x,u16 y)函数。

以上四个函数是LCD的主要函数,是最底层的。其他任何功能的函数都可以在这几个底层函数基础上实现。其他功能的LCD驱动函数均在tftlcd.c里面有定义和说明,具体见附件。

VS1003模块驱动程序设计

VS1003也是采用SPI模式,不过是挂在SPI1上面,这里主要介绍VS1003的初始化操作。在对MCU相关IO口正确配置之后就可以对VS1003模块进行初始化了。VS1003通过7根线与MCU通信: XRST、XDCS、XCS、DREQ、SCK、SO、SI。

XRST:VS1003复位线,低电平有效。XDCS:数据片选信号,低电平有效。XCS:命令片选信号,低电平有效。DREQ:数据请求,输入总线。SCK、SI、SO:SPI接口线。

VS1003与MCU的通讯都是通过SPI总线来完成的,在默认情况下,数据将在SCLK的上升沿有效(被读入VS1003),一次需要在SCLK的下降沿更新数据,并且字节发送以MSB在先。注意VS1003的最大写入和读出时钟分别是CLKI/4和CLKI/6(CLKI为VS1003内部时钟)。

VS1003模块初始化步骤:

硬复位,XRST =0;延时,XDCS、XCS、XRST置1;等待DREQ为高;软件复位:SPI_MODE=0X0804;等待DREQ为高(软件复位结束);设置VS1003的时钟:SCI_CLOCKF=0X9800,3倍频;设置VS1003的采样率:SPI_AUDATA=0XBB81,采样率48K,立体声;设置重音:SPI_BASS=0X0055;设置音量:SCI_VOL=0X;向VS1003发送四个字节无效数据,启动SPI发送;

VS1003的初始化在VS1003x.c里面,通过void Vs1003_Init(void)函数实现。

TEA5767模块驱动程序设计

TEA5767收音机模块支持IIC和三线模式,这里我们使用IIC来控制。TEA5767的器件地址是0XC0,在对TEA5767的读操作通过写入0XC1来执行。

TEA5767写操作:

发送IIC起始信号发送器件地址0XC0等待应答发送一个字节,等待应答,再发送一个字节,等待应答,循环5次发送IIC停止信号

TEA5767的读操作与写操作基本相同,只是IIC开始之后写入0XC1,将发送一个字节改为接收一个字节就可以了。关于TEA5767的其他操作函数均在TEA5767.c里面

5 部分核心代码

#include "vs1003.h" //VS1003的全功能函数//支持SIN测试和RAM测试//并加入了VS1003的频谱显示代码,不过说实话不咋地,还不如自己写的频谱分析,怀疑是不是真实的频谱变换? //正点原子@SCUT//V1.1//VS1003设置参数//0,henh.1,hfreq.2,lenh.3,lfreq 5,主音量u8 vs1003ram[5]={0,0,0,0,250};//保存VS1003的设置//EEPROM地址:486~490 共五个void Save_VS_Set(void){u8 t;for(t=0;t<5;t++)FM24C16_WriteOneByte(488+t,vs1003ram[t]);//vs1003ram保存 }//读取VS1003的设置//EEPROM地址:486~490 共五个void Read_VS_Set(void){u8 t;for(t=0;t<5;t++)vs1003ram[t]=FM24C16_ReadOneByte(488+t);//vs1003ram调用} //SPI1口读写一个字节//TxData:要发送的字节//返回值:读取到的字节u8 SPI1_ReadWriteByte(u8 TxData){while((SPI1->SR&1<<1)==0);//等待发送区空 SPI1->DR=TxData; //发送一个byte while((SPI1->SR&1<<0)==0);//等待接收完一个byte return SPI1->DR;//返回收到的数据 } //设置SPI1的速度//SpeedSet:1,高速;0,低速;void SPI1_SetSpeed(u8 SpeedSet){SPI1->CR1&=0XFFC7;if(SpeedSet==1)//高速{SPI1->CR1|=6<<3;//Fsck=Fpclk/64=1.125Mhz}else//低速{SPI1->CR1|=6<<3; //Fsck=Fpclk/128=562.5Khz}SPI1->CR1|=1<<6; //SPI设备使能 } //软复位VS1003void Vs1003SoftReset(void){u8 retry; while((GPIOC->IDR&MP3_DREQ)==0);//等待软件复位结束SPI1_ReadWriteByte(0X00);//启动传输retry=0;while(Vs1003_REG_Read(SPI_MODE)!=0x0804)// 软件复位,新模式 {Vs1003_CMD_Write(SPI_MODE,0x0804);// 软件复位,新模式delay_ms(2);//等待至少1.35ms if(retry++>100)break; } while ((GPIOC->IDR & MP3_DREQ) == 0);//等待软件复位结束 retry=0;while(Vs1003_REG_Read(SPI_CLOCKF)!=0X9800)//设置vs1003的时钟,3倍频 ,1.5xADD {Vs1003_CMD_Write(SPI_CLOCKF,0X9800);//设置vs1003的时钟,3倍频 ,1.5xADDif(retry++>100)break; } retry=0;while(Vs1003_REG_Read(SPI_AUDATA)!=0XBB81)//设置vs1003的时钟,3倍频 ,1.5xADD {Vs1003_CMD_Write(SPI_AUDATA,0XBB81);if(retry++>100)break; }//Vs1003_CMD_Write(SPI_CLOCKF,0X9800);//Vs1003_CMD_Write(SPI_AUDATA,0XBB81); //采样率48k,立体声 set1003();//设置VS1003的音效 ResetDecodeTime();//复位解码时间 //向vs1003发送4个字节无效数据,用以启动SPI发送MP3_DCS_SET(0);//选中数据传输SPI1_ReadWriteByte(0XFF);SPI1_ReadWriteByte(0XFF);SPI1_ReadWriteByte(0XFF);SPI1_ReadWriteByte(0XFF);MP3_DCS_SET(1);//取消数据传输delay_ms(20);} //硬复位MP3void Mp3Reset(void){MP3_RST_SET(0);delay_ms(20);MP3_DCS_SET(1);//取消数据传输MP3_CCS_SET(1);//取消数据传输MP3_RST_SET(1); while((GPIOC->IDR & MP3_DREQ)==0);//等待DREQ为高delay_ms(20); }//正弦测试 void VsSineTest(void){Mp3Reset(); Vs1003_CMD_Write(0x0b,0X); //设置音量 Vs1003_CMD_Write(SPI_MODE,0x0820);//进入vs1003的测试模式 while ((GPIOC->IDR & MP3_DREQ) == 0);//等待DREQ为高//向vs1003发送正弦测试命令:0x53 0xef 0x6e n 0x00 0x00 0x00 0x00//其中n = 0x24, 设定vs1003所产生的正弦波的频率值,具体计算方法见vs1003的datasheetMP3_DCS_SET(0);//选中数据传输SPI1_ReadWriteByte(0x53);SPI1_ReadWriteByte(0xef);SPI1_ReadWriteByte(0x6e);SPI1_ReadWriteByte(0x24);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);delay_ms(100);MP3_DCS_SET(1); //退出正弦测试MP3_DCS_SET(0);//选中数据传输SPI1_ReadWriteByte(0x45);SPI1_ReadWriteByte(0x78);SPI1_ReadWriteByte(0x69);SPI1_ReadWriteByte(0x74);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);delay_ms(100);MP3_DCS_SET(1); //再次进入正弦测试并设置n值为0x44,即将正弦波的频率设置为另外的值MP3_DCS_SET(0);//选中数据传输SPI1_ReadWriteByte(0x53);SPI1_ReadWriteByte(0xef);SPI1_ReadWriteByte(0x6e);SPI1_ReadWriteByte(0x44);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);delay_ms(100);MP3_DCS_SET(1);//退出正弦测试MP3_DCS_SET(0);//选中数据传输SPI1_ReadWriteByte(0x45);SPI1_ReadWriteByte(0x78);SPI1_ReadWriteByte(0x69);SPI1_ReadWriteByte(0x74);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);delay_ms(100);MP3_DCS_SET(1); } //ram 测试 void VsRamTest(void){u16 regvalue ; Mp3Reset();Vs1003_CMD_Write(SPI_MODE,0x0820);// 进入vs1003的测试模式while ((GPIOC->IDR&MP3_DREQ)==0); // 等待DREQ为高MP3_DCS_SET(0); // xDCS = 1,选择vs1003的数据接口SPI1_ReadWriteByte(0x4d);SPI1_ReadWriteByte(0xea);SPI1_ReadWriteByte(0x6d);SPI1_ReadWriteByte(0x54);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);SPI1_ReadWriteByte(0x00);delay_ms(50); MP3_DCS_SET(1);regvalue=Vs1003_REG_Read(SPI_HDAT0); // 如果得到的值为0x807F,则表明完好。printf("regvalueH:%x\n",regvalue>>8);//输出结果 printf("regvalueL:%x\n",regvalue&0xff);//输出结果 }//向VS1003写命令//address:命令地址//data:命令数据void Vs1003_CMD_Write(u8 address,u16 data){while((GPIOC->IDR&MP3_DREQ)==0);//等待空闲SPI1_SetSpeed(0);//低速 MP3_DCS_SET(1); //MP3_DATA_CS=1;MP3_CCS_SET(0); //MP3_CMD_CS=0; SPI1_ReadWriteByte(VS_WRITE_COMMAND);//发送VS1003的写命令SPI1_ReadWriteByte(address); //地址SPI1_ReadWriteByte(data>>8); //发送高八位SPI1_ReadWriteByte(data); //第八位MP3_CCS_SET(1);//MP3_CMD_CS=1; SPI1_SetSpeed(1);//高速} //向VS1003写数据void Vs1003_DATA_Write(u8 data){MP3_DCS_SET(0); //MP3_DATA_CS=0;SPI1_ReadWriteByte(data);MP3_DCS_SET(1); //MP3_DATA_CS=1;MP3_CCS_SET(1); //MP3_CMD_CS=1; } //读VS1003的寄存器 //读VS1003//注意不要用倍速读取,会出错u16 Vs1003_REG_Read(u8 address){u16 temp=0; while((GPIOC->IDR&MP3_DREQ)==0);//非等待空闲状态 SPI1_SetSpeed(0);//低速 MP3_DCS_SET(1); //MP3_DATA_CS=1;MP3_CCS_SET(0); //MP3_CMD_CS=0;SPI1_ReadWriteByte(VS_READ_COMMAND);//发送VS1003的读命令SPI1_ReadWriteByte(address); //地址temp=SPI1_ReadWriteByte(0xff);//读取高字节temp=temp<<8;temp+=SPI1_ReadWriteByte(0xff); //读取低字节MP3_CCS_SET(1);//MP3_CMD_CS=1; SPI1_SetSpeed(1);//高速return temp; } //FOR WAV HEAD0 :0X7761 HEAD1:0X7665 //FOR MIDI HEAD0 :other info HEAD1:0X4D54//FOR WMA HEAD0 :data speed HEAD1:0X574D//FOR MP3 HEAD0 :data speed HEAD1:ID//比特率预定值const u16 bitrate[2][16]={{0,8,16,24,32,40,48,56,64,80,96,112,128,144,160,0}, {0,32,40,48,56,64,80,96,112,128,160,192,224,256,320,0}};//返回Kbps的大小//得到mp3&wma的波特率u16 GetHeadInfo(void){unsigned int HEAD0;unsigned int HEAD1; HEAD0=Vs1003_REG_Read(SPI_HDAT0); HEAD1=Vs1003_REG_Read(SPI_HDAT1);switch(HEAD1){case 0x7665:return 0;//WAV格式case 0X4D54:return 1;//MIDI格式 case 0X574D://WMA格式{HEAD1=HEAD0*2/25;if((HEAD1%10)>5)return HEAD1/10+1;else return HEAD1/10;}default://MP3格式{HEAD1>>=3;HEAD1=HEAD1&0x03; if(HEAD1==3)HEAD1=1;else HEAD1=0;return bitrate[HEAD1][HEAD0>>12];}} } //重设解码时间void ResetDecodeTime(void){Vs1003_CMD_Write(SPI_DECODE_TIME,0x0000);Vs1003_CMD_Write(SPI_DECODE_TIME,0x0000);//操作两次}//得到mp3的播放时间n secu16 GetDecodeTime(void){return Vs1003_REG_Read(SPI_DECODE_TIME); } //加载频谱分析的代码到VS1003void LoadPatch(void){u16 i;for (i=0;i<943;i++)Vs1003_CMD_Write(atab[i],dtab[i]); delay_ms(10);}//得到频谱数据void GetSpec(u8 *p){u8 byteIndex=0;u8 temp;Vs1003_CMD_Write(SPI_WRAMADDR,0x1804); for (byteIndex=0;byteIndex<14;byteIndex++) {temp=Vs1003_REG_Read(SPI_WRAM)&0x63;//取小于100的数 *p++=temp;} }void SPI1_RST(void){RCC->APB2RSTR|=1<<12;//复位SPI1delay_ms(10); RCC->APB2RSTR&=~(1<<12);//结束复位SPI1delay_ms(10); SPI1->CR1|=0<<10;//全双工模式SPI1->CR1|=1<<9; //软件nss管理SPI1->CR1|=1<<8; SPI1->CR1|=1<<2; //SPI主机SPI1->CR1|=0<<11;//8bit数据格式SPI1->CR1|=1<<1; //空闲模式下SCK为1 CPOL=1SPI1->CR1|=1<<0; //数据采样从第二个时间边沿开始,CPHA=1 SPI1->CR1|=6<<3; //Fsck=Fpclk/128 =562.5khzSPI1->CR1|=0<<7; //MSBfirst SPI1->CR1|=1<<6; //SPI设备使能} //设定vs1003播放的音量和高低音 void set1003(void){u8 t;u16 bass=0; //暂存音调寄存器值u16 volt=0; //暂存音量值u8 vset=0; //暂存音量值 vset=255-vs1003ram[4];//取反一下,得到最大值,表示最大的表示 volt=vset;volt<<=8;volt+=vset;//得到音量设置后大小//0,henh.1,hfreq.2,lenh.3,lfreq for(t=0;t<4;t++){bass<<=4;bass+=vs1003ram[t]; }Vs1003_CMD_Write(SPI_BASS,bass);//BASS Vs1003_CMD_Write(SPI_VOL,volt); //设音量 } //初始化VS1003的IO口 void Vs1003_Init(void){RCC->APB2ENR|=1<<2; //PORTA时钟使能 RCC->APB2ENR|=1<<12;//SPI1时钟使能 //存储器映射,不用理 #ifdef VECT_TAB_RAMNVIC_SetVectorTable(NVIC_VectTab_RAM, 0x0); #else NVIC_SetVectorTable(NVIC_VectTab_FLASH, 0x0); #endif GPIOA->CRL&=0X000FFFFF;//PA5.6.7复用输出GPIOA->CRL|=0XBBB00000; GPIOA->ODR|=0X00E0;//PA5.6.7上拉 SPI1->CR1|=0<<10;//全双工模式SPI1->CR1|=1<<9; //软件nss管理SPI1->CR1|=1<<8; SPI1->CR1|=1<<2; //SPI主机SPI1->CR1|=0<<11;//8bit数据格式SPI1->CR1|=1<<1; //空闲模式下SCK为1 CPOL=1SPI1->CR1|=1<<0; //数据采样从第二个时间边沿开始,CPHA=1 SPI1->CR1|=6<<3; //Fsck=Fpclk/128 =562.5khzSPI1->CR1|=0<<7; //MSBfirst SPI1->CR1|=1<<6; //SPI设备使能//以上初始化VS1003的SPI连接口,SPI1口//所以上拉之前,必须使能时钟.才能实现真正的上拉输出RCC->APB2ENR|=1<<2; //PA时钟使能RCC->APB2ENR|=1<<4; //PC时钟使能RCC->APB2ENR|=1<<0; //开启辅助时钟AFIO->MAPR=0X04000000;//关闭JTAG,只有关闭JTAG,才能使用PA14 GPIOA->CRH&=0XF0FFFFF0;//PA.8/14推挽输出GPIOA->CRH|=0X03000003; GPIOA->ODR|=0X4100; //上拉GPIOC->CRH&=0XFFFFFF00;GPIOC->CRH|=0X00000083;//PC.8输出 ,PC9输入GPIOC->ODR|=1<<8;//PC.8上拉 GPIOC->ODR|=1<<9; //PC.9上拉 }

6 最后

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。