STM32 串口烧写 FLASH 外部字库 UCGUI显示 自我学习总结
扫描二维码
随时随地手机看文章
最近学习TFT显示问题,在多种汉字显示方面有点难,主要是字库太大,几个字库就不得了。开始是使用SD卡向外部FLASH---W25X16写,完全能够完成。后来觉得这样比较麻烦,有时候还没有SD接口,于是打算用串口写一下试一试,网上有很多人说会丢失数据,在后面的试验中暂时没有发现。
我是在我前段时间学习的UCGUI的基础基础上修改的。
主要功能是----启动开发板,首先写入地址指令----必须十六进制----比如---2A 23000500 0323 2A-----其中2A,23为验证码,前后都有,第3,4位是地址码,前面就表示将要写入的起始地址是0x05*4096,第5,6位是为了写入数据将要从起始地址连续擦除的扇区个数,前面就表示0x0003个。
--------------主函数-------------------
int main(void)
{
uint16_t Erase_flag=0,i=0;
uint32_t Erase_adrr=0;
bsp_InitLCD();
bsp_InitTouch();
SPI_GPIO_Init(4);
NVIC_Configuration();
USART1_Config();
GPIO_Other_Config();
//TIM3_Init(84,1000); //实时检测触摸屏信息
GUI_Init();
GUI_SetBkColor(GUI_RED); //背景颜色
GUI_Clear();//用背景颜色清屏
GUI_CURSOR_Show(); //显示光标---大小32
GUI_SetFont(&GUI_Font32B_1); //设置字体
GUI_SetColor(GUI_BLACK); //设置文本颜色
GUI_DispStringAt("Addr:",10,50); //指定位置显示指定文本
GUI_DispStringAt("Sect:",10,85);
GUI_DispStringAt("Nums:",10,120);
GUI_DispStringAt("Eras:",10,160);
GUI_DispStringAt("Test:",10,200);
while(1)
{
CK_flag=0; //串口接收数据是否是地址的的标识判断位---为0 表示后面接收的是地址
RS232_RX_CNT=0;//串口接收到地址数据时有作用,每次到这里串口接收缓存数组下表为0,表示后面将接收地址
addr_i=0; //串口接收到非地址数据的个数,它的变化实在串口接收中断函数中递增,这里清零表示再次烧写
Delay_Ms(1000);
GUI_SetFont(&GUI_Font32B_1);
GUI_SetColor(GUI_YELLOW);
GUI_DispStringAtCEOL("Wait Send Addr",50,10);
GUI_DispStringAt("Waits Addr",100,160);
GUI_DispHexAt(addr_Stop,200,50,6);//烧录完成后根据真实烧录的字节数计算出终止地址显示出来
//下面这个while循环,其目的是等待串口接收地址指令---指令需要16进制发送---比如2A 23 00 50 00 01 23 2A---为一个完整指令
//2A 23 都为判断标识---第3,4表示起始地址--0x05*4096---第5,6数据表示从起始地址需要擦除多少个扇区来方便后面写数据--0x0002个
while(!(RS232_RX_BUF[0]==0x2A)||!(RS232_RX_BUF[1]==0x23)||!(RS232_RX_BUF[6]==0x23)||!(RS232_RX_BUF[7]==0x2A));
addr=RS232_RX_BUF[2]*256+RS232_RX_BUF[3]; //addr为接收到的起始地址所在扇区,也就是说是第几个扇区
addr_Stop =addr*4096;//第几个扇区对应的扇区地址
Erase_adrr =addr; //起始地址也就是要擦除的扇区是第几个
Erase_flag=RS232_RX_BUF[4]*256+RS232_RX_BUF[5];//要擦出的连续扇区个数
GUI_SetFont(&GUI_Font32B_1);
GUI_SetColor(GUI_CYAN);
GUI_DispHexAt(addr*4096,100,50,6);//显示起始地址
GUI_DispDecAt(Erase_flag,100,85,4);//显示要擦出的扇区个数
GUI_DispStringAt("Start Erase",100,160);
for(i=0;i
Erase_adrr=addr+i;//扇区地址自增---这里自己试验的时候不小心把addr写成了Erase_adrr,导致调试好久,最后一步一步添加验证才找到问题,不过在这个过程中加深了自己对FLASH的了解!
SPI_FLASH_SectorErase(4096*Erase_adrr);//擦出扇区
GUI_SetColor(GUI_WHITE);
GUI_DispDecAt(i+1,255,160,4);// 显示实时擦除扇区状态---第几个
Delay_Ms(50);
}
GUI_SetColor(GUI_CYAN);
GUI_DispStringAt("Finis Erase",100,160);
GUI_SetFont(&GUI_FontHZ16);
GUI_SetColor(GUI_DARKMAGENTA);
GUI_DispStringAt("啊波次的乐我许与在 #*aA/字库",100,210);//这里是用来测试前面的擦出是否完成---比如前面擦出了ASCII码的字库地址,其中的FLASH就不会显示
CK_flag=10;//串口接收数据是否是地址的的标识判断位 ---为10 表示后面接收的是数据
GUI_SetFont(&GUI_Font32B_1);
GUI_SetColor(GUI_GREEN);
GUI_DispStringAtCEOL("Wait Send Data",50,10);
while(GPIO_ReadInputDataBit(GPIOA,GPIO_Pin_1) == 0); //这里等于是一个开关---用来等待串口传输完成,可以再串口上看到是否传输完成
GUI_SetColor(GUI_WHITE);
GUI_DispDecAt(addr_i,100,120,7);
GUI_SetFont(&GUI_FontHZ16);
GUI_SetColor(GUI_BLUE);
GUI_DispStringAt("啊波次的乐我许与在 #*aA/字库",100,210);//这里是用来测试前面的烧写是否完成---比如前面擦出了ASCII码的字库地址这里又重新烧录了ASCII码,其中的FLASH就会再次显示
Delay_Ms(100);
GUI_TOUCH_Exec();
GUI_Exec();
}
}
--------------串口中断函数------------
void USART1_IRQHandler(void)
{
uint8_t res;
if(USART_GetITStatus(USART1, USART_IT_RXNE) != RESET)//接收到数据
{
USART_ClearITPendingBit(USART1, USART_IT_RXNE);
res=USART_ReceiveData(USART1); //读取接收到的数据
RS232_RX_BUF[RS232_RX_CNT++]=res;//接收数据,主要保存地址指令
if(CK_flag==10) //串口接收数据
{
addr_i++; //烧写地址自增---主函数中将在烧写完成后显示
SPI_FLASH_BufferWrite(&res,addr++,1); //烧写接收到的数据到FLASH
}
if( RS232_RX_CNT==12)//无影响
RS232_RX_CNT=0;
}
}
---------------------实验实物图片----------------------
----标识显示-----
-----指令---擦除0x000000起始的连续64个扇区
----上面擦除的是汉字库的内容---故下面汉字没有显示---
----重新烧录汉字库,一共261696个字节数据---大约64K---
-----烧录完成,汉字又显示了----
----指令----擦除0x050000起始的一个扇区---
-----擦除了ASCII码地址内容,故ASCII码没有显示--
-----重新烧录ASCII码库,一个4096个字节数据,刚好4K---
----烧录完成,ASCII码再次显示----