RISC-V MCU CH32V103將常量定義到指定的Flash地址


RISC-V MCU將常量定義到指定的Flash地址 -- 以CH32V103為例

Keil MDK開發ARM 內核的MCU時,將常量定義到指定的Flash地址中,使用 _attribute_( at(絕對地址) )即可,如:

const u32 myConstVariable_1[128] __attribute__((at(0x08001000))) = {0x12345678,0x22221111};//定位在flash中,其他flash補充為0 

沁恆RISC-V MCU ,通過Mounriver Studio(MRS)開發時,暫時不支持_attribute_( at(絕對地址) )命令。可通過如下步驟實現:

1、編輯ld鏈接文件,添加SECTIONS段

	.flash_test_address :
	{
		. = ALIGN(4);              /*4字節對齊*/
		. = ORIGIN(FLASH)+0x1000;  /*ORIGIN(FLASH)為 MEMORY定義的FLASH的起始地址(CH32V103為0x08000000),指定到從FLASH起始的0x1000長度的位置*/
		KEEP(*(SORT_NONE(.test_address_1)))  /*鏈接時*KEEP()可以使得被標記段的內容不被清除*/
		. = ALIGN(4);
	} >FLASH AT>FLASH 

如需將變量定義到Flash的最后,將此段添加到 .text段后面,注意指定的Flash地址要大於程序編譯大小。

2、函數中使用__attribute__((section(".xxx")))定義常量

2.1 定義單字節常量

const uint8_t myConstVariable_1 __attribute__((section(".test_address_1"))) = 0x11;/*地址為0x00001000*/

查看map文件,常量地址如下:

image

sections .flash_test_address段中以4字節對齊,其余3字節補0。

二進制bin文件0x1000地址信息如下;

image

2.2 定義連續的多個單字節常量

const uint8_t myConstVariable_1 __attribute__((section(".test_address_1"))) = 0x11; /*地址為0x00001002*/
const uint8_t myConstVariable_2 __attribute__((section(".test_address_1"))) = 0x22; /*地址為0x00001001*/
const uint8_t myConstVariable_3 __attribute__((section(".test_address_1"))) = 0x33; /*地址為0x00001000*/

ld文件中flash_test_address 段默認從指定地址開始為其分配連續的地址,查看map文件,常量地址如下:

image

二進制bin文件0x1000地址信息如下;

image

2.3 定義多個不連續的常量

此時需要修改ld文件

.flash_test_address :
	{
		. = ALIGN(4);              /*4字節對齊*/
		. = ORIGIN(FLASH)+0x1000;  /*ORIGIN(FLASH)為 MEMORY定義的FLASH的起始地址(CH32V103為0x08000000),指定到從FLASH起始的0x1000長度的位置*/
		KEEP(*(SORT_NONE(.test_address_1)))  /*鏈接時*KEEP()可以使得被標記段的內容不被清除*/
		. = ORIGIN(FLASH)+0x1040;  /*ORIGIN(FLASH)為 MEMORY定義的FLASH的起始地址(CH32V103為0x08000000),指定到從FLASH起始的0x1040長度的位置*/
		KEEP(*(SORT_NONE(.test_address_2)))  /*鏈接時*KEEP()可以使得被標記段的內容不被清除*/
		. = ALIGN(4);
	} >FLASH AT>FLASH 

在函數中定義兩個指定地址的常量

const uint8_t myConstVariable_1[8] __attribute__((section(".test_address_1"))) = {0x11,0x22,0x33,0x44}; /*首地址為0x00001000*/
const uint8_t myConstVariable_2[4] __attribute__((section(".test_address_2"))) = {0x55,0x66}; /*首地址為0x00001040*/

查看map文件,常量地址如下:

image

二進制bin文件0x1000地址信息如下;

image

這樣指定的方式會造成中間段有56個字節的flash無法分配內容,浪費了,不建議這樣指定,如果實在要這樣做,需要嚴格控把控,可根據間隔的大小,指定編譯后小於該間隔的函數存儲到該flash塊。

如指定函數Delay_Init編譯后存放test_address_1塊內,緊跟定義的常量后。

/*******************************************************************************
* Function Name  : Delay_Init
* Description    : Initializes Delay Funcation.
* Input          : None
* Return         : None
*******************************************************************************/
__attribute__((section(".test_address_1"))) void Delay_Init(void)
{
	p_us=SystemCoreClock/8000000;
	p_ms=(uint16_t)p_us*1000;
}

Delay_Init函數編譯后的大小為0x2a,編譯后的map文件如下:

image

二進制bin文件0x1000地址信息如下:
image
(新增的A2 4A 04 指令暫時不詳)


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM