娇小w搡bbbb搡bbb,《第一次の人妻》,中国成熟妇女毛茸茸,边啃奶头边躁狠狠躁视频免费观看

歷史上的今天

今天是:2025年02月15日(星期六)

2020年02月15日 | STM32_IAP詳解(有代碼,有上位機)

發布者:polkmm 來源: elecfans關鍵字:STM32  IAP  上位機 手機看文章 掃描二維碼
隨時隨地手機看文章

Iap,全名為in applacation programming,即在應用編程,與之相對應的叫做isp,in system programming,在系統編程,兩者的不同是isp需要依靠燒寫器在單片機復位離線的情況下編程,需要人工的干預,而iap則是用戶自己的程序在運行過程中對User Flash 的部分區域進行燒寫,目的是為了在產品發布后可以方便地通過預留的通信口對產品中的固件程序進行更新升級。在工程應用中經常會出現我們的產品被安裝在某個特定的機械結構中,更新程序的時候拆機很不方便,使用iap技術能很好地降低工作量.


實現iap有兩個很重要的前提,首先,單片機程序能對自身的內部flash進行擦寫,第二,單片機要有能夠和外部進行通訊的方式,無論是網絡還是別的方式,只要能傳輸數據就行


通常實現 IAP 功能時,即用戶程序運行中作自身的更新操作,需要在設計固件程序時編寫兩個項目代碼,第一個項目程序不執行正常的功能操作,而只是通過某種通信方式(如 USB、 USART)接收程序或數據,執行對第二部分代碼的更新;第二個項目代碼才是真正的功能代碼。這兩部分項目代碼都同時燒錄在 User Flash 中,當芯片上電后,首先是第一個項目代碼開始運行,它作如下操作: 

1)檢查是否需要對第二部分代碼進行更新 

2)如果不需要更新則轉到 4) 

3)執行更新操作 

4)跳轉到第二部分代碼執行 


第一部分代碼必須通過其它手段,如 JTAG 或 ISP 燒入;第二部分代碼可以調用第一部分的功能


也就是說,將iap和app做成兩個程序,這是其中的一種策略,還有一種策略,可以把iap程序和app程序做在一個代碼中,但是那樣耦合性有點高,我們先進行第一種嘗試.


要做iap首先我們要知道stm32的啟動流程,流程如下

1、單片機從0x80000000位置啟動,并將該地址當成系統棧頂地址

2、運行到中斷向量表中,默認的中斷向量表為0x80000004,該位置存放復位中斷

3、跳轉到復位中斷處理函數當中,進行系統初始化,然后運行main函數


當我們準備用iap的時候,單片機內部是有著兩套程序的,這個時候我們就需要在iap中

和app中分別放置兩套中斷向量表,當iap代碼中將app燒寫到flash中之后,跳轉到app的中斷向量表中,程序就可以正常執行了,當然需要修改某些系統設置,使得在app和iap階段單片機可見的中斷向量表只能有一套(具體請查看stm32芯片的啟動代碼)


而當需要從app跳轉到iap的時候,只需要將app的中斷向量表修改成iap的中斷向量表,同時主動跳轉到iap的reset中斷處理程序,這樣就能再次開始iap流程.


這樣,在系統中就需要我們確定幾個東西,第一個是iap程序的中斷向量表,為0x80000004位置(80000000存放的是msp的初始值),第二個是app程序的中斷向量表,該位置需要根據iap程序的長度計算,比如iap占用了64K,那么512K的芯片而言,就還有448K的空間存放app程序,448K的最開始放置中斷向量表,位置就應該是0x08000000+0x10004的位置.


Cortex-m3的中斷向量并不是在程序中固定的,我們可以通過修改某些寄存器來修改對于當前應用的中斷向量表位置.


決定中斷向量表的寄存器是如下這個

通過修改這個寄存器的值,我們可以控制對于當前單片機應用來說可見的向量表的位置(也就說說邏輯上我們有兩個向量表,但是同一時間只有一個運行)


以上是內核階段的操作,在此之外我們還需要對stm32的flash進行編程,那么就涉及到刪除的編程和擦除操作,這需要參考stm32的閃存編程手冊


首先,當單片機復位之后,閃存式被鎖住的,需要主動去解鎖,向FLASH_KEYR寫入兩個指定的連續鍵值用于解鎖


然后將需要寫入的閃存擦除,擦除之后在進行寫入,寫入完成,再次上鎖


對應的代碼如下

u16 STMFLASH_BUF[STM_SECTOR_SIZE/2];//最多是2K字節

void STMFLASH_Write(u32 WriteAddr,u16 *pBuffer,u16 NumToWrite)

{

u32 secpos;       //扇區地址

u16 secoff;       //扇區內偏移地址(16位字計算)

u16 secremain; //扇區內剩余地址(16位字計算)      

u16 i;   

u32 offaddr;   //去掉0X08000000后的地址

if(WriteAddr=(STM32_FLASH_BASE+1024*STM32_FLASH_SIZE)))return;//非法地址

FLASH_Unlock();                     //解鎖

offaddr=WriteAddr-STM32_FLASH_BASE;            //實際偏移地址.

secpos=offaddr/STM_SECTOR_SIZE;        //扇區地址  0~127 for STM32F103RBT6

secoff=(offaddr%STM_SECTOR_SIZE)/2;    //在扇區內的偏移(2個字節為基本單位.)

secremain=STM_SECTOR_SIZE/2-secoff;    //扇區剩余空間大小  

if(NumToWrite<=secremain)secremain=NumToWrite;//不大于該扇區范圍

while(1)

{  

STMFLASH_Read(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//讀出整個扇區的內容

for(i=0;i

{

if(STMFLASH_BUF[secoff+i]!=0XFFFF)break;//需要擦除    

}

if(i

{

FLASH_ErasePage(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE);//擦除這個扇區

for(i=0;i

{

STMFLASH_BUF[i+secoff]=pBuffer[i];    

}

STMFLASH_Write_NoCheck(secpos*STM_SECTOR_SIZE+STM32_FLASH_BASE,STMFLASH_BUF,STM_SECTOR_SIZE/2);//寫入整個扇區 

}else STMFLASH_Write_NoCheck(WriteAddr,pBuffer,secremain);//寫已經擦除了的,直接寫入扇區剩余區間.                  

if(NumToWrite==secremain)break;//寫入結束了

else//寫入未結束

{

secpos++;              //扇區地址增1

secoff=0;              //偏移位置為0    

pBuffer+=secremain;    //指針偏移

WriteAddr+=secremain;  //寫地址偏移   

NumToWrite-=secremain; //字節(16位)數遞減

if(NumToWrite>(STM_SECTOR_SIZE/2))secremain=STM_SECTOR_SIZE/2;//下一個扇區還是寫不完

else secremain=NumToWrite;//下一個扇區可以寫完了

}   

}; 

FLASH_Lock();//上鎖

該函數可以實現flash的寫入操作,接下來我們需要定義一套通訊協議用于串口數據傳輸

//串口接收緩沖區

u8 serial_Buffer[SERIAL_MAX_LENGTH] = {0};

//串口接收數據長度

u16 serial_Buffer_Length = 0;

u8 receiveMode = 0;//接收參數的中斷處理模型,為0的時候是命令模式,為1的時候為下載模式

u8 receiveExpectCount = 0;//串口期望接收長度

//串口中斷處理

static void SerialRecv(u8 ch)

{

if(receiveMode == 0)

{

if((serial_Buffer_Length&0x8000) == 0x8000)//已經接收完成,系統還沒處理

{

serial_Buffer_Length |= 0x8000;//退出

}

else if((serial_Buffer_Length&0x4000) == 0x4000)//接收到回車還沒接收到換行

{

if(ch == 'n')serial_Buffer_Length |= 0x8000;

else

{

//一幀接受失敗

serial_Buffer_Length = 0;

}

}

else

{

if((serial_Buffer_Length&0xff) < SERIAL_MAX_LENGTH)

{

if(ch == 'r')serial_Buffer_Length |= 0x4000;

else

{

serial_Buffer[(serial_Buffer_Length&0xff)] = ch;

serial_Buffer_Length++;

}

}

else

{

//一幀接受失敗

serial_Buffer_Length = 0;

}

}

}

else

{

//下載模式,只控制字符串的量,數據的第一位是該數據包的長度,接收到這么多長度,接收完成位置一

//注意,在這種模式下,清除serial_Buffer_Length之前應當清除receiveExpectCount的值

if(receiveExpectCount == 0)//期望下載為0,第一個數就是期望下載數

{

receiveExpectCount = ch;

}

else

{

if((serial_Buffer_Length&0x8000) == 0x8000)//已經接收完成,系統還沒處理,此時不接收數據

{

serial_Buffer_Length |= 0x8000;//退出

}

else

{

serial_Buffer[(serial_Buffer_Length&0xff)] = ch;//接收數據并保存

serial_Buffer_Length++;

if((serial_Buffer_Length&0xff) == receiveExpectCount)//接收到了期望長度的數據

{

serial_Buffer_Length |= 0x8000;//一包接收完成標志

}

}

}

}

}

這樣系統就能接收數據了,接下來定義五個命令

"iap_down"

"iap_jump_app"

"iap_over"

"iap_set_flag"

"iap_clear_flag"

第一個命令為系統開始下載,在這個命令之后上位機就能夠將程序數據發下來了,

第二個命令為iap跳轉到app的跳轉指令

第三個命令是指示iap完成,將系統緩沖區清空的指令

第四個指令為設置app標志,當iap檢測到該標志的時候直接跳轉到app程序中

第五個指令為清除app標志,讓iap程序不自動跳轉到app程序中,我們分別來看

首先是iap_set_flag,其響應函數如下

#define APP_CONFIG_ADDR     0X08001FFC //配置地址

#define APP_CONFIG_SET_VALUE    0X5555 //設置值

#define APP_CONFIG_CLEAR_VALUE  0XFFFF //清零值

//設置app固化配置

void iap_set_flag_s(void)

{

Test_Write(APP_CONFIG_ADDR,APP_CONFIG_SET_VALUE);

printf("okrn");

}

我們使用0x08000000-0x08003000來存放iap代碼,并將0X08001FFC作為存放app固化標志的地方

//清除app固化配置

void iap_clear_flag(void)

{

Test_Write(APP_CONFIG_ADDR,APP_CONFIG_CLEAR_VALUE);

printf("okrn");

}

對iap_jump2app命令的響應如下

//跳轉到應用程序段

//appxaddr:用戶代碼起始地址.

void iap_load_app(u32 appxaddr)

{

if(((*(vu32*)appxaddr)&0x2FFE0000)==0x20000000)  //檢查棧頂地址是否合法.0x20000000是sram的起始地址,也是程序的棧頂地址

{

printf("okrn");

Delay_Ms(10);

jump2app=(iapfun)*(vu32*)(appxaddr+4);    //用戶代碼區第二個字為程序開始地址(復位地址)     

MSR_MSP(*(vu32*)appxaddr);                //初始化APP堆棧指針(用戶代碼區的第一個字用于存放棧頂地址)

jump2app();                               //跳轉到APP.

}

else

{

printf("program in flash is errorrn");

}

}

//跳轉到app區域運行

void iap_jump_app_s(void)

{

iap_load_app(FLASH_APP1_ADDR);//跳轉到app的復位向量地址

}

接下來就是iap_down,用于下載的核心算法

#define FLASH_APP1_ADDR     0x08002000     //第一個應用程序起始地址(存放在FLASH)

//保留的空間為IAP使用

u16 iapbuf[1024] = {0}; //用于緩存數據的數組

u16 receiveDataCur = 0;  //當前iapbuffer中已經填充的數據長度,一次填充滿了之后寫入flash并清零

u32 addrCur = FLASH_APP1_ADDR;         //當前系統寫入地址,每次寫入之后地址增加2048

//開始下載

void iap_down_s(void)

{

u16 i = 0;

u16 temp = 0;

u16 receiveCount;

printf("begin,wait data downloadrn");

receiveMode = 1;//串口進入下載接收數據模式

while(1)

{

//循環接收數據,每次必須要發128個數據下來,如果沒有128,說明這是最后一包數據

//接收到一包數據之后,返回一個小數點,發送完成,系統編程完成之后返回一個iap_over

if(serial_Buffer_Length & 0x8000)

{

receiveCount = (u8)(serial_Buffer_Length&0x00ff);

if(receiveCount == 128)//滿足一包,填充并查看是否有了1024字節,有了寫入閃存

{

for(i = 0; i < receiveCount; i+=2)

{

//數據八位融合為16位

temp = (((u16)serial_Buffer[i+1])<<8) + ((u16)serial_Buffer[i]);

iapbuf[receiveDataCur] = temp;

receiveDataCur++;//完成之后receiveDataCur++;

}

receiveExpectCount = 0;//清除期望接收模式

serial_Buffer_Length = 0;//清除串口滿標志

printf(".");//每次接受一次數據打一個點

//此時需要檢測receiveDataCur的值,要是放滿了,就需要寫入

if(receiveDataCur == 1024)

{

//寫入flash中

STMFLASH_Write(addrCur,iapbuf,1024);

//printf("rnwrite addr %x,length 1024rn",addrCur);

addrCur += 2048;//地址+2048

//寫完之后receiveDataCur要清零等待下一次傳輸

receiveDataCur = 0;

}

else //有可能最后一包有128個數據但是最終沒有2048個數據,此時擴展一個指令用于完成最后一個的寫入

{

}

//還沒放滿,等待下一次數據過來

}

else   //不滿足一包,說明數據傳送這是最后一包,寫入閃存

{

//沒有一包也要傳送到緩存中

for(i = 0; i < receiveCount; i+=2)

{

//數據八位融合為16位

temp = (((u16)serial_Buffer[i+1])<<8) + ((u16)serial_Buffer[i]);

iapbuf[receiveDataCur] = temp;

receiveDataCur++;//完成之后receiveDataCur++;

}

receiveExpectCount = 0;//清除期望接收模式

serial_Buffer_Length = 0;//清除串口滿標志

printf(".");//每次接受一次數據打一個點

//之后就要將這數據寫入到閃存中

STMFLASH_Write(addrCur,iapbuf,receiveDataCur);//將最后的一些內容字節寫進去.

//printf("rnwrite addr %x,length %drn",addrCur,receiveDataCur);

//寫完之后要把地址恢復到原來的位置

addrCur = FLASH_APP1_ADDR;

receiveDataCur = 0;

//寫完之后要退出下載循環并告訴上位機,已經下載完了

printf("download overrn");

//同時,也要退出下載循環模式

receiveMode = 0;

return;

}

這段代碼的核心思想是上位機每次發送128個數據下來,發滿了2048個寫一次flash,當最后一包數據不是128的時候說明數據發送完成了,這時候退出下載模式,但是當遇到最后一包數據也是128個時候怎么辦呢,于是定義了這個指令

iap_over,上位機偵測到最后一包數據也是128個的時候補充發送該命令,下位機將緩存寫入并退出

//最后一包有128個數據但是最終沒有2048個數據

//收到這個指令檢測receiveDataCur和addrCur的值,

//完成最后的寫入

void iap_over_s(void)

{

//這個時候,依然在串口下載模式

if(receiveDataCur != 0)

{

STMFLASH_Write(addrCur,iapbuf,receiveDataCur);//將最后的一些內容字節寫進去.

//printf("write addr %x,length %d",addrCur,receiveDataCur);

addrCur = FLASH_APP1_ADDR;

receiveDataCur = 0;

//同時,也要退出下載模式

receiveMode = 0;

}

printf("okrn");

}

這是iap的核心代碼,接下來我們在main函數中檢測app固化標志,如果標志位設置,那么跳轉到app

if(STMFLASH_ReadHalfWord(APP_CONFIG_ADDR) == 0x5555)

{

//直接跳轉到APP

iap_jump_app_s();

}

到這里基本上就完成了iap的工作,可是想想,還需要設置一個地方,我們要在target中設置使用的flash空間,不能超范圍,如下

如果需要flash下載的話還需要設置jlink的flash下載設置如下.

這樣可以直接使用jlink將代碼下載到單片機中,而且不會影響原先的app程序,注意,要選擇erase sector used,不能全部擦除flash

橋斗麻袋,我們忘了一件事情,假設我們設置了app標志,那及時app能跳轉到iap中,iap豈不是馬上會跳轉回app,永遠不能等待下載?

解決辦法就是我們在app中app跳轉到iap的指令中將app固化標志清除掉,在app代碼中添加一條指令

Iap,其響應方法為

__asm void MSR_MSP(u32 addr)

{

MSR MSP, r0           //set Main Stack value

BX r14

}

void iap_jump(u32 iapxaddr)

{

if(((*(vu32*)iapxaddr)&0x2FFE0000)==0x20000000)  //檢查棧頂地址是否合法.0x20000000是sram的起始地址,也是程序的棧頂地址

{

printf("okrn");

Delay_Ms(10);

jump2iap=(iapfun)*(vu32*)(iapxaddr+4);    //用戶代碼區第二個字為程序開始地址(復位地址)     

[1] [2]
關鍵字:STM32  IAP  上位機 引用地址:STM32_IAP詳解(有代碼,有上位機)

上一篇:單片機STM32時鐘圖文理解
下一篇:STM32內存管理以及STM32中的堆棧

推薦閱讀

  人工智能是目前新興科技中最炙手可熱的項目之一。中共十九大召開以后,中國進一步加速人工智能的研發,有望成為世界人工智能第一大國。  力爭成人工智能第一大國  法國《論壇報》指出,中國許多新興科技企業都已經越過了起步階段,力爭成為人工智能領域的佼佼者。  報道稱,為了實現這一目標,中國可謂盡心盡力。近日,中國政府宣布將在北京西部...
三星作為世界出貨量最大的手機制造商自然不會只有Galaxy S/Note這兩個系列,大部分的銷量還是憑借A系列這個中低端產品的。在三星將A系列交給OEM廠商后希望能在成本上進一步降低,給消費者們更加優惠的選擇,其中即將發布的A10、A30、A50都將采用塑料機身,產品定位上A10更為低端。  A10將搭載6.2英吋的Infinity-V屏幕,這種屏幕類似于水滴屏...
/********************************************************************1602液晶顯示********************************************************************/#include <msp430x14x.h>#define CPU_F ((double)8000000) #define delay_us(x) __delay_cycles((long)(CPU_F*(double)x/1000000.0)) #define delay_ms(x) __delay_cycles((long)(CPU_F*...
外媒 mspoweruser 報道,蘋果 iPhone 13(或稱 iPhone 12S)系列將在 6 個多月后到來,但據說由于在家辦公的原因,蘋果公司一直很難做好新 iPhone 的保密工作。 在最新的視頻中,EverythingApplePro 匯總了 iPhone 13 系列所有最新的傳聞,包括一些獨家項目爆料。 其中最令人意想不到的一條源自 Max Weinbach,據說蘋果終于要為 iPhone...

史海拾趣

問答坊 | AI 解惑

20元打造經典PC遙控器(2)

圖三 兩個重要元件的引腳示意圖               首先,要切割一塊合適的電路板。這里我用多孔的萬用板,我做小東西時通常都是使用它,很方便,不過做復雜一點就不太好了。當然有條件可以自已加工印 ...…

查看全部問答∨

VC.60和VxWorks5.0下程序執行效率測試的問題

我現在做了個編碼工具,簡單的說就是把源程序(標準C寫的)經過編碼后變成另外一個程序(也是C代碼),使程序的安全性得到提高,當然運行速度會變慢。 但現在我遇到的問題是:我在VC.60和VxWorks5.0下分別做了一個效率測試,測試結果如下: 在VC6 ...…

查看全部問答∨

怎么用KIEL燒錄UPSD的程序

  我以前一直是用PSDSOFT EXPRESS+ FLASHLINK燒程序的。最近換了個筆記本沒有并口了, 在網上看到FLASHLINK有ULINK 和RLINK的替代品 但是RLINK我所在的城市買不到,ULINK不支持PSDSOFT EXPRESS環境下燒錄。   進入正題  ...…

查看全部問答∨

求助!!iar 中從mcs51類型到msp430 類型

小弟看mcs51 的代碼,__bit __no__init bool a;如果這個語法從for mcs 51 的iar 中轉到for msp430的iar中該怎么寫啊,這個語法也不太懂啊,我該從哪方面下手 啊?請大俠們幫忙!!!謝謝了 !!…

查看全部問答∨

Wince 開發討論

  本人是嵌入式軟件開發網的站長,先自我介紹一下,我叫Mark,從事WinCE的裁剪,底層開發,相關產品應用開發已經有一段時間了,今年上半年我建立了一個QQ群,群名是:WinCE之家,群號是:36278326。想與搞WinCE的同仁一起探討相關的開發 ...…

查看全部問答∨

申請ez430-RF2500 開發板

  [ 本帖最后由 shzps 于 2010-11-24 21:40 編輯 ]…

查看全部問答∨

求dsPIC33FJ系列中文資料

主要是想要dsPIC33FJ256中文資料。…

查看全部問答∨

負載電阻的問題

最近學別人的電路,看到這樣的電路圖 我問這個電阻有什么用,他說是負載電阻,然后就沒了...............哎..... 所以上來問問大家,這個電阻到底有什么用呢?GND一般都是接在大地或者機器外殼上,那么這個P_GND是接到哪里的呢,我在電路圖里沒 ...…

查看全部問答∨

請問哪位仁兄有SP12的data sheet

請問哪位仁兄有SP12的data sheet,本人由于設計的需要,要用到它。請有此資料的仁兄高抬貴手,給我發個來,在下不勝感激。我的郵箱:tsm234@163.com。…

查看全部問答∨
小廣播
設計資源 培訓 開發板 精華推薦

最新單片機文章
何立民專欄 單片機及嵌入式寶典

北京航空航天大學教授,20余年來致力于單片機與嵌入式系統推廣工作。

 
EEWorld訂閱號

 
EEWorld服務號

 
汽車開發圈

 
機器人開發圈

電子工程世界版權所有 京ICP證060456號 京ICP備10001474號-1 電信業務審批[2006]字第258號函 京公網安備 11010802033920號 Copyright ? 2005-2025 EEWORLD.com.cn, Inc. All rights reserved
主站蜘蛛池模板: 高雄市| 博野县| 乌拉特后旗| 凌海市| 屏山县| 恩施市| 姚安县| 灯塔市| 高雄县| 平顶山市| 大埔县| 大庆市| 延寿县| 浦江县| 吉安市| 安庆市| 延津县| 响水县| 富阳市| 大名县| 曲麻莱县| 喜德县| 资阳市| 内乡县| 迁西县| 会泽县| 巨野县| 故城县| 堆龙德庆县| 江北区| 茂名市| 都昌县| 峨边| 正安县| 周宁县| 鹿泉市| 贞丰县| 北宁市| 灌阳县| 高淳县| 精河县|