KEIL環境下如何讓代碼在 RAM中運行(轉)

原文鏈接:https://www.stmcu.org.cn/article/id-328210?=edm

KEIL環境下如何讓代碼在 RAM中運行

發佈時間:2016-08-26

來源:ST社區

標籤:KEILNucleoSTM32

分享到:

 


 

前言
經常遇到有人使用KEIL時需要將部分或者全部程序代碼放到RAM中運行的問題,現將其總結在本文中。通過STM32F411Nucleo的一個例子來介紹幾種讓程序在RAM中運行的方法。

我們先從ToggleLED函數在Flash中執行亮滅開始。下面是ToggleLED函數和它的調用情況。在main函數的while(1)裏調用ToggleLED。

void ToggleLED(void)
{ HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_5); /* Insert a 100ms delay */
HAL_Delay(100);
}

int main(void)
{ …… /*##-3- Toggle PA05 IO in an infinite loop ######*/
while (1) { ToggleLED(); }
}

編譯環境的Linker的配置見下圖:
Flash起始地址:0x08000000
RAM起始地址:0x20000000

1

 

編譯後從map文件可以看到,ToggleLED以及其中調用到的HAL_GPIO_TogglePin和HAL_Delay函數的地址都在FLASH中。

2
將翻轉LED的程序放到SRAM中執行
方法一:通過#pragma arm section code = “RAMCODE ”和#pragma arm section。參考Example1代碼。
這種方式,可以同時將多個函數放到指定的section。具體方法如下:
1. 修改.sct文件,自定義一個叫做RAMCODE的section,放在RW_IRAM1執行區域,地址範圍0x20000000~0x20020000。

LR_IROM1 0x08000000 0x00080000 { ; load region size_region
ER_IROM1 0x08000000 0x00080000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$$Sections)
.ANY (+RO)
} RW_IRAM1 0x20000000 0x00020000 { ; RW data
*.o(RAMCODE)
.ANY (+RW +ZI)
}
}

2. 在工程中使用前面修改的.sct文件

3

3.以#pragma arm section code = “RAMCODE” 開頭,以#pragma arm section結尾。將所有需要放到RAMCODE section的函數包括進來。編譯時,編譯器會自動將這些函數放到RAMCODE所在0x20000000開始的區域。

#pragma arm section code = "RAMCODE"
void ToggleLED(void)
{ HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_5); /* Insert a 100ms delay */
HAL_Delay(100);
}

void HAL_GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{ /* Check the parameters */
assert_param(IS_GPIO_PIN(GPIO_Pin));
GPIOx->ODR ^= GPIO_Pin;
}

uint32_t HAL_GetTick(void)
{ return tick;
}

void HAL_Delay(__IO uint32_t Delay)
{ uint32_t tickstart = 0;
tickstart = HAL_GetTick();
while((HAL_GetTick() - tickstart) < Delay)
{
}
}
#pragma arm section

4.從map文件裏,可以看到這四個函數都已經被放到了SRAM中。

4

方法二:通過__attribute__((section(“name ”)))
在KEIL中可以通過__attribute__((at(address)))的方式將變量放到指定的位置。
通過__attribute__((section(“name ”)))的方式將變量或者函數放到指定的位置。

下面我們來看看如何通過這種方式將程序放到SRAM中執行。
1.同樣,我們需要修改.sct文件,自定義一個叫做RAMCODE的section,並在工程選項的linker頁面中,選擇定義好的.sct文件。(見方法一中的第1,2步)

LR_IROM1 0x08000000 0x00080000 { ; load region size_region ER_IROM1 0x08000000 0x00080000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$$Sections)
.ANY (+RO)
}
RW_IRAM1 0x20000000 0x00020000 { ; RW data
*.o(RAMCODE)
.ANY (+RW +ZI)
}
}

2.在需要放到RAM中的函數前,用__attribute__((section("RAMCODE")))聲明該函數放在RAMCODE section中。注意,該函數中調用到的所有函數也要放到RAMCODE section中。

__attribute__((section("RAMCODE")))
void ToggleLED(void)
{ HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_5); /* Insert a 100ms delay */
HAL_Delay(100);
}

__attribute__((section("RAMCODE")))
void HAL_GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{ /* Check the parameters */
assert_param(IS_GPIO_PIN(GPIO_Pin));
GPIOx->ODR ^= GPIO_Pin;
}

__attribute__((section("RAMCODE")))
__weak uint32_t HAL_GetTick(void)
{ return uwTick; }

__attribute__((section("RAMCODE")))
__weak void HAL_Delay(__IO uint32_t Delay)
{ uint32_t tickstart = 0;
tickstart = HAL_GetTick();
while((HAL_GetTick() - tickstart) < Delay)
{ }
}

3.從編譯後的map文件可以看出,ToggleLED以及它調用到的所有函數都被到了RAM中。

00

方法二可以覆蓋方法一,也就是說如果你同時用方法一和方法二對同一個函數的執行區域做了說明。最終起作用的是方法二。還是通過上面提到的代碼來說明。
修改.sct文件。將SRAM分爲兩個執行區RW_IRAM1和RW_IRAM2。Section RAMCODE1,RAMCODE2分別位於0x20000000開始,和0x20010000開始的兩個64KB的區域。

LR_IROM1 0x08000000 0x00080000 { ; load region size_region ER_IROM1 0x08000000 0x00080000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$$Sections)
.ANY (+RO)
}

RW_IRAM1 0x20000000 0x00010000 { ; RW data
*.o(RAMCODE1)
.ANY (+RW +ZI)
}

RW_IRAM2 0x20010000 0x00010000 {
*.o(RAMCODE2) }
}

2.在代碼中, HAL_GetTick被放在了#pragma的作用域內被聲明放在RAMCODE1 section,同時又用__attribute__( ( section ( "RAMCODE2" ) ) ) 將其放在RAMCODE2的section內。

#pragma arm section code = "RAMCODE1"
void ToggleLED(void)
{ HAL_GPIO_TogglePin(GPIOA, GPIO_PIN_5);
/* Insert a 100ms delay */
HAL_Delay(100); }

void HAL_GPIO_TogglePin(GPIO_TypeDef* GPIOx, uint16_t GPIO_Pin)
{ /* Check the parameters */
assert_param(IS_GPIO_PIN(GPIO_Pin));
GPIOx->ODR ^= GPIO_Pin;
}

__attribute__( ( section ( "RAMCODE2" ) ) )
uint32_t HAL_GetTick(void)
{ return tick; }

void HAL_Delay(__IO uint32_t Delay)
{ uint32_t tickstart = 0;
tickstart = HAL_GetTick();
while((HAL_GetTick() - tickstart) < Delay) { }
}
#pragma arm section

3.編譯完成後,我們看看map文件中HAL_GetTick被放到了哪個section。

5

從map裏可以看到,最終HAL_GetTick被放在了RAMCODE2 section中。

**如何將整個程序放到SRAM中執行
前面介紹了將一個或多個程序放到指定地址執行的方法。如果需要放到指定地址的程序比較多,我們還可以將這些需要放到指定地址的程序集中放到一個或幾個C文件中,然後在.sct文件中將這些C文件生成的目標文件放到指定地址。
在這裏,我們將嘗試將整個程序放到SRAM中執行。復位後程序從FLASH啓動,之後將從SRAM執行所有的程序。下面是具體的步驟。
1.將中斷向量表和中斷處理程序放到SRAM中
新建一個startup_stm32f411xe_ram.s文件,放到0x20000000開始的位置(在.sct文件中修改)。注意這裏是新建,而不是直接將原來的文件放到SRAM中,爲什麼呢?大家可以思考一下。在startup_stm32f411xe_ram.s裏定義新的SECTION,叫做RESET_ram(還有其他的修改,請對照參考代碼)。在後面的.sct中將把RESET_ram這個section放到SRAM開始的位置上(見第3步)。

Vector Table Mapped to Address 0 at Reset
AREA RESET_ram, DATA, READONLY
EXPORT __Vectors_ram
EXPORT __Vectors_End_ram
EXPORT __Vectors_Size_ram
__Vectors_ram DCD 0 ; Top of Stack
DCD 0 ; Reset Handler
DCD NMI_Handler ; NMI Handler
……

2.在SystemInit中將中斷向量表的偏移地址設置爲0x20000000。使能VECT_TAB_SRAM。

#ifdef VECT_TAB_SRAM
SCB->VTOR = SRAM_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal SRAM */
#else SCB->VTOR = FLASH_BASE | VECT_TAB_OFFSET; /* Vector Table Relocation in Internal FLASH */
#endif

3.修改.sct文件,將運行時需要的所有目標文件都放到SRAM執行區中。這裏中斷向量表有同樣的兩份,一份在0x08000000開始的位置,一份在0x20000000開始的位置。

LR_IROM1 0x08000000 0x00080000 { ; load region size_region
ER_IROM1 0x08000000 0x00080000 { ; load address = execution address
*.o (RESET, +First)
*(InRoot$$Sections)
.ANY (+RO)
}
RW_IRAM1 0x20000000 0x00020000 { ; RW data
*.o (RESET_ram, +First)
startup_stm32f411xe_ram.o(+RO)
main.o(+RO +RW)
stm32f4xx_it.o(+RO +RW)
stm32f4xx_hal.o(+RO +RW)
stm32f4xx_hal_gpio.o(+RO +RW)
stm32f4xx_hal_rcc.o(+RO +RW)
stm32f4xx_hal_cortex.o(+RO +RW)
.ANY (+RW +ZI)
}
}

4. 編譯完成後,從map文件或者跟蹤調試的結果都可以看到。系統復位以後,從main函數開始,所有的程序都在RAM中運行了。
另外,如果你的程序中有用到ARM底層的庫,可以在.sct文件中加入*armlib*(+RO)來將所有用到的庫文件放到SRAM中。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章