找回密碼
 立即注冊(cè)

QQ登錄

只需一步,快速開始

搜索
查看: 2530|回復(fù): 1
收起左側(cè)

使用STM32F103單片機(jī)片內(nèi)FLASH虛擬U盤,用于存儲(chǔ)文件IAP升級(jí)

[復(fù)制鏈接]
ID:219315 發(fā)表于 2023-6-11 09:52 | 顯示全部樓層 |閱讀模式
   最近做一個(gè)傳感器的項(xiàng)目,要求在傳感器里存儲(chǔ)校正參數(shù),主機(jī)連接不同的傳感器時(shí)都能準(zhǔn)確采值。于是想到用片內(nèi)FLASH程序存儲(chǔ)空間劃一部分出來,虛擬成一個(gè)U盤存儲(chǔ)校正參數(shù),可以節(jié)省一片EEROM芯片。網(wǎng)上找了很久,不是不能用,就是要收費(fèi),這里發(fā)出來方便有需要的人。虛擬U盤部分已經(jīng)測(cè)試通過,后面IAP部分因?yàn)橛貌坏,沒有測(cè)試。

利用片內(nèi)FLASH虛擬U盤,使用文件復(fù)制方式實(shí)現(xiàn)IAP
USART IAP功能想必大家都很熟悉了,我們通過USART  IAP,可以很方便的實(shí)現(xiàn)程序升級(jí)。
但是在實(shí)際工作中,遇到一些特殊情況時(shí),USART  IAP功能卻不像我們應(yīng)用中那么方便,甚至
成了一大制約因素:
1.板子交給遠(yuǎn)方的用戶,用戶手頭沒有帶串口的電腦。
2.用戶找到帶串口的電腦,卻仍然沒有合適的串口連接線。
3.終于找到了合適的串口連接線,仍沒有IAP軟件。
4.IAP軟件連同HEX文件一同發(fā)給用戶,用戶卻不會(huì)操作。在我們手頭很簡(jiǎn)單的操作,到用戶手
里,卻總也搞不定(即便是簡(jiǎn)單的超級(jí)終端,想要遠(yuǎn)程教會(huì)用戶,也是很考驗(yàn)功力的)。
5.不是所有的板子都設(shè)計(jì)有串口,沒有串口的板子,呵呵…… 幸好STM還給我們提供了DFU升級(jí)方式,基本可以解決以上多數(shù)問題,只需要留出USB接口就行。
但是DFU方式,仍然對(duì)用戶的要求很高,需要用戶熟悉軟件的操作過程和流程(見制約因素4)。
只有找到一種更簡(jiǎn)單的方式,只要簡(jiǎn)單的復(fù)制、粘貼操作,就可以實(shí)現(xiàn)程序更新,才能最大限度解決以上問題。
現(xiàn)在我們通過虛擬U盤IAP程序,將程序存儲(chǔ)的FLASH空間虛擬為U盤,只要將程序文件復(fù)制到這個(gè)U盤,就完成了程序更新,消除了用戶升級(jí)操作的學(xué)習(xí)過程。
51hei.jpg
0x8000000
0x8003000
0x8008000
無IAP程序  一般IAP程序  U盤IAP程序
文件系統(tǒng)
IAP程序  IAP程序
用戶程序  用戶程序  用戶程序 程序?qū)崿F(xiàn)過程:
1.IAP程序
IAP程序仍然占用0x8000000~0x8003000的FLASH空間,IAP的作用,是在滿足IAP控制要求的情況下啟動(dòng)IAP功能,否則,直接跳轉(zhuǎn)并執(zhí)行用戶程序。啟動(dòng)IAP功能,可以使用3個(gè)方式:
1.1.當(dāng)用戶程序?yàn)榭諘r(shí),自動(dòng)啟動(dòng)IAP功能,做寫程序準(zhǔn)備。
1.2.當(dāng)用戶程序正常,但啟動(dòng)時(shí)按下了特定的IAP按鍵時(shí),啟動(dòng)IAP功能。
1.3.用戶程序運(yùn)行中,在某個(gè)特殊位置做了IAP升級(jí)標(biāo)志,判讀標(biāo)志后,啟動(dòng)IAP功能。
現(xiàn)在的IAP程序,僅僅實(shí)現(xiàn)簡(jiǎn)單的將內(nèi)部FLASH虛擬為U盤功能即可。參見《5分鐘用STM32的內(nèi)置Flash做一個(gè)超小U盤》。當(dāng)然,這個(gè)小U盤缺少了文件系統(tǒng),需要重新格式化才行。 格式化后,我們的超小U盤OK了,接下來就該進(jìn)行下一步了,用戶程序。
2.用戶程序
2.1.配置
常規(guī)的用戶程序,對(duì)于我們的STM32F103C8系列,有64KFLASH,因此可供IAP程序使用的為0x8003000~0x800FFFF的FLASH空間。但我們這次的用戶程序,必須要保留下文件系統(tǒng),因此修改為僅使用0x8008000~0x800FFFF。這其中的0X8003000~0X8008000空間,就是格式化時(shí)保存的文件系統(tǒng)。
51hei.png
為何修改為0x8008000而不是其它位置?因?yàn)橄到y(tǒng)告訴我們了,U盤容量=32K,因此反推:
(0x800ffff+1)-32K=0x8008000(STM32F103C8系列)。  
51hei.png
2.2.編譯用戶程序
51hei.png
主程序中不要忘記偏移。編譯完,趕緊試試吧?墒,我們?cè)搶⒛膫(gè)文件存入U(xiǎn)盤。
51hei.png
這樣就會(huì)編譯結(jié)束后自動(dòng)生成對(duì)應(yīng)的***.BIN文件,這個(gè)BIN文件,就是我們需要的升級(jí)文件。 2.3.復(fù)制文件,升級(jí)了。復(fù)制文件后復(fù)位,用戶程序自動(dòng)運(yùn)行了。

這時(shí),我們可以再次回到U盤狀態(tài),刪除文件,甚至是格式化這個(gè)U盤,只要沒有新的文件進(jìn)行覆蓋,我們的程序會(huì)始終正常運(yùn)行的。這是FAT表文件系統(tǒng)的缺陷造成的,也是數(shù)據(jù)恢復(fù)的理論基礎(chǔ)。

STM32單片機(jī)源程序如下:
  1. #include "stm32f10x.h"
  2. #include "stdio.h"
  3. #include "hw_config.h"
  4. #include "usb_lib.h"
  5. #include "usb_pwr.h"

  6. extern uint16_t MAL_Init (uint8_t lun);


  7. typedef  void (*pFunction)(void);
  8. #define ApplicationAddress 0x0800A800  //計(jì)算跳轉(zhuǎn)地址512K-470K = 42K 即0xA800  ApplicationAddress = 0x8000000+0xA800 = 0x0800A800
  9. uint32_t JumpAddress;
  10. pFunction Jump_To_Application;

  11. int main(void)
  12. {
  13.         GPIO_InitTypeDef GPIO_InitStructure;        
  14.         RCC_APB2PeriphClockCmd(RCC_APB2Periph_GPIOB, ENABLE);        
  15.         GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;   
  16.         GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
  17.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_Out_PP;
  18.         GPIO_Init(GPIOB, &GPIO_InitStructure);
  19.         GPIO_ResetBits(GPIOB, GPIO_Pin_9);
  20.         
  21.         GPIO_InitStructure.GPIO_Pin  = GPIO_Pin_8;
  22.         GPIO_InitStructure.GPIO_Mode = GPIO_Mode_IPU;
  23.         GPIO_Init(GPIOB, &GPIO_InitStructure);
  24.         
  25.         
  26. //        if(GPIO_ReadInputDataBit(GPIOB,GPIO_Pin_8)==0)
  27. //        {
  28.                 Set_System();
  29.                 Set_USBClock();
  30.                 USB_Interrupts_Config();
  31.                 USB_Init();
  32.                 while (bDeviceState != CONFIGURED);
  33.                         while(1)
  34.         {
  35.                
  36.         }

  37. //        }

  38.                 __set_PRIMASK(1);
  39.                 if(((*(__IO uint32_t *) ApplicationAddress) & 0x2FFE0000) == 0x20000000)
  40.                                 {
  41.                                         JumpAddress = *(__IO uint32_t *) (ApplicationAddress + 4);
  42.                                         Jump_To_Application = (pFunction) JumpAddress;
  43.                                        
  44.                                         __set_MSP(*(__IO uint32_t *) ApplicationAddress);
  45.                                         Jump_To_Application();
  46.                                 }
  47.         while(1)
  48.         {
  49.                
  50.         }

  51. }        

復(fù)制代碼

Keil代碼下載: STM32F103RET6.7z (373.32 KB, 下載次數(shù): 38)

評(píng)分

參與人數(shù) 1黑幣 +50 收起 理由
admin + 50 共享資料的黑幣獎(jiǎng)勵(lì)!

查看全部評(píng)分

回復(fù)

使用道具 舉報(bào)

ID:766804 發(fā)表于 2023-12-19 11:00 | 顯示全部樓層
512K-470K = 42K這個(gè)470K怎么來的?
回復(fù)

使用道具 舉報(bào)

本版積分規(guī)則

手機(jī)版|小黑屋|51黑電子論壇 |51黑電子論壇6群 QQ 管理員QQ:125739409;技術(shù)交流QQ群281945664

Powered by 單片機(jī)教程網(wǎng)

快速回復(fù) 返回頂部 返回列表