1、STM32 启动文件与 .sct 文件分析

1) 定义STACK段,{NOINIT,读写}:分配一段内存大小为0.5K;

2) 定义HEAP段, {NOINIT,读写}:分配一段内存大小为1K;

3) 定义RESET段,{DATA,只读}:DCD各种中断向量;

4) 定义|.text|段,{CODE,只读}:Reset_Handler函数,函数中最后加载了__main;

对剩余的中断函数进行了弱定义;

在最后还有一段用户初始化堆栈的代码__user_initial_stackheap。

那这些代码都存放在什么位置呢?

5) 分析 .sct 文件:

分散加载文件(即scatter file,后缀为.scf)。

分散加载文件是一个文本文件,通过编写一个分散加载文件来指定ARM连接器在生成映像文件时如何分配RO,RW,ZI等数据的存放地址。

如果不用SCATTER文件指定,那么ARM连接器会按照默认的方式来生成映像文件,一般情况下我们是不需要使用分散加载文件的。

但在某些场合,我们希望把某些数据放在指定的地址处,那么这时候SCATTER文件就发挥了非常大的作用。

而且SCATTER文件用起来非常简单好用。

举个例子:

比如像LPC2378芯片具有多个不连续的SRAM,通用的RAM是32KB,可是32KB不够用,我想把某个.C中的RW数据放在USB的SRAM中,那么就可以通过SCATTER文件来完成这个功能。

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

.ANY (+RW +ZI)

}

}

STACK段和HEAP段是RW属性,存在RAM(0x20000000-0x20010000)中,具体的地址由编译器在后面链接时决定,并不是一定存在RAM的开头地址。

RESET段存在FLASH(0x08000000-0x08080000)中,而且是FLASH的最开头,再结合CORTEX-M3的特性,其上电后根据启动引脚来决定PC位置,比如启动设置为FLASH启动,则启动后PC跳到0x08000000。

此时CPU会先取2个地址(硬件决定),第一个是栈顶地址,第二个是复位异常地址,这样就跳到Reset_Handler,Reset_Handler执行到将最后跳转到ç库的__main。

|.text |段是CODE属性,也存在FLASH区。

启动代码所做的工作如下:

先是建立了堆栈,之后上电后寻找到中断向量表中的复位函数Reset_Handler执行,之后跳转到__main执行Ç库函数,最后由__main调用main()函数,进入C的世界。

2、__user_initial_stackheap

这段代码位于裸机启动文件的末尾:

IF      :DEF:__MICROLIB

EXPORT  __initial_sp

EXPORT  __heap_base

EXPORT  __heap_limit

ELSE

IMPORT  __use_two_region_memory

EXPORT  __user_initial_stackheap

__user_initial_stackheap

LDR     R0, =  Heap_Mem

LDR     R1, =(Stack_Mem + Stack_Size)

LDR     R2, = (Heap_Mem +  Heap_Size)

LDR     R3, = Stack_Mem

BX      LR

若是使用了microlib,则只需要将__initial_sp,__ heap_base的,__ heap_limit三个变量定义成全局变量即可(这三个变量也是固定的可被Ç库引用,在库中需要使用到这三个变量对堆栈进行初始化);

否则,就需要自己定义__user_initial_stackheap。

microlib缺省的情况下使用的是Keil C库。

但是事实上,μVision库里包含了更多__user_initial_stackheap()的函数体,这样编译器可以根据开发人员scatter文件的内容自动选择合适的函数体。

换句话说,针对RVCT v3.x及之后的版本,使用scatter文件的开发人员可以不再重新实现__user_initial_stackheap()的函数体。

也就是说不必再自己写了__user_initial_stackheap,自己在做实验验证时,没有使用microlib库,同时也将这部分函数注释掉,并没有产生任何异常。

所以对__user_initial_stackheap在这里就不再做更多深入的研究了,这一部分太烧脑了,就当作C库已经为我们准备好了这个函数。

3、堆栈的单区模型和双区模型

堆栈分为单区模型和双区模型:

单区模型堆和栈在同一存储器区中互相朝向对方增长

双区模型将堆和栈分别放置在存储器不同的区中,__ user_initial_stackheap()建立的专用堆限制来检查堆,需要设置堆栈的长度。

1)选择使用单区模型,在SCT文件中定义一个特殊的执行域,使用符号:

ARM_LIB_STACKHEAP,并使用EMPTY属性这样库管理器就选择了一个把这个域当作堆和栈合并在一起的__user_initial_stackheap()函数。

在这个函数中使用了“Image$ $ ARM_LIB_STACKHEAP$ $Base”和“Image$ $ARM_LIB_STACKHEAP$ $ZI$ $Limit”符号。

2)选择使用双区模型,在sct文件中定义两个特殊的执行域,使用符号:

ARM_LIB_STACK和ARM_LIB_HEAP,都要使用EMPTY属性。这样库管理器就会选择使用符号:“Image$ $ARM_LIB_HEAP$ $Base” ,“Image$ $ARM_LIB_STACK$ $ZI$ $ limit”,“Image$ $ARM_LIB_STACK$ $Base”,“Image$ $ARM_LIB_STACK$ $ZI$ $Limit”的__user_initial_stackheap()函数。

从裸机的启动文件可以看出,裸机使用的是双区模型。

4、Huawei_LiteOS 启动文件与 sct 文件

启动文件:

LOS_Heap_Min_Size   EQU     0x400

AREA    LOS_HEAP, NOINIT, READWRITE, ALIGN=3

__los_heap_base

LOS_Heap_Mem    SPACE   LOS_Heap_Min_Size

AREA    LOS_HEAP_INFO, DATA, READONLY, ALIGN=2

IMPORT  |Image$$ARM_LIB_STACKHEAP$$ZI$$Base|

EXPORT  __LOS_HEAP_ADDR_START__

EXPORT  __LOS_HEAP_ADDR_END__

__LOS_HEAP_ADDR_START__

DCD     __los_heap_base

__LOS_HEAP_ADDR_END__

DCD     |Image$$ARM_LIB_STACKHEAP$$ZI$$Base| - 1

PRESERVE8

AREA    RESET, CODE, READONLY

THUMB

IMPORT  ||Image$$ARM_LIB_STACKHEAP$$ZI$$Limit||

IMPORT  osPendSV

EXPORT  _BootVectors

EXPORT  Reset_Handler

_BootVectors

DCD     ||Image$$ARM_LIB_STACKHEAP$$ZI$$Limit||

DCD     Reset_Handler

Reset_Handler

IMPORT  SystemInit

IMPORT  __main

LDR     R0, =SystemInit

BLX     R0

LDR     R0, =__main

BX      R0

ALIGN

END

定义LOS_HEAP段,{NOINIT,读写}:分配一段内存大小为1K;

定义LOS_HEAP_INFO段,{DATA,只读}:定义__LOS_HEAP_ADDR_START__和__LOS_HEAP_ADDR_END__这两个全局变量供OS使用;

定义RESET段,{CODE,只读}:启动向量,第一个是栈顶地址,第二个是Reset_Handler;将Reset_Handler主体也写入了RESET段;

首先,可以看出,分配堆栈的方式与裸机不同,使用的是单区模型,从下向上排列

__LOS_HEAP_ADDR_START __ = __ los_heap_base,为堆低地址;

__LOS_HEAP_ADDR_END __ = |Image$ $ARM_LIB_STACKHEAP$ $ZI$ $Base | - 1,为堆顶(不确定的地址);

|Image$ $ARM_LIB_STACKHEAP$ $ZI$ $Base|,为栈底(不确定的地址);

|Image$ $ARM_LIB_STACKHEAP$ $ZI$ $Limit|,为栈顶地址;

所以,由上文可知,在sct文件中必然会出现ARM_LIB_STACKHEAP这个执行域:

LR_IROM1 0x08000000 0x00020000  {    ; load region size_region

ER_IROM1 0x08000000 0x00020000  {    ; load address = execution address

*.o (RESET, +First)

*(InRoot$$Sections)

.ANY (+RO)

* (LOS_HEAP_INFO)

}

VECTOR 0x20000000 0x400  {    ; Vector

* (.data.vector)

}

RW_IRAM1 0x20000400 0x00004800  {    ; RW data

;.ANY (+RW +ZI)

* (.data, .bss)

* (LOS_HEAP)

}

ARM_LIB_STACKHEAP 0x20004C00 EMPTY 0x400  {    ;LiteOS MSP

}

}

那么其他的异常中断向量入口在哪里呢?定义在los_hwi.c文件中被定义成了数组的形式:

#ifdef __ICCARM__

#pragma  location = ".data.vector"

#elif defined (__CC_ARM) || defined (__GNUC__)

LITE_OS_SEC_VEC

#endif

HWI_PROC_FUNC m_pstHwiForm[OS_VECTOR_CNT] =

{

(HWI_PROC_FUNC)0,                    // [0] Top of Stack

(HWI_PROC_FUNC)Reset_Handler,        // [1] reset

(HWI_PROC_FUNC)osHwiDefaultHandler,  // [2] NMI Handler

(HWI_PROC_FUNC)osHwiDefaultHandler,  // [3] Hard Fault Handler

(HWI_PROC_FUNC)osHwiDefaultHandler,  // [4] MPU Fault Handler

(HWI_PROC_FUNC)osHwiDefaultHandler,  // [5] Bus Fault Handler

(HWI_PROC_FUNC)osHwiDefaultHandler,  // [6] Usage Fault Handler

(HWI_PROC_FUNC)0,                    // [7] Reserved

(HWI_PROC_FUNC)0,                    // [8] Reserved

(HWI_PROC_FUNC)0,                    // [9] Reserved

(HWI_PROC_FUNC)0,                    // [10] Reserved

(HWI_PROC_FUNC)osHwiDefaultHandler,  // [11] SVCall Handler

(HWI_PROC_FUNC)osHwiDefaultHandler,  // [12] Debug Monitor Handler

(HWI_PROC_FUNC)0,                    // [13] Reserved

(HWI_PROC_FUNC)osPendSV,             // [14] PendSV Handler

(HWI_PROC_FUNC)osHwiDefaultHandler,  // [15] SysTick Handler

};

这一部分代码被分散加载文件加载到了VECTOR段,位于RAM的开头部分。

在内核初始化时会进行中断向量表重映射的工作。

keil的sct文件_STM32 分散加载文件 .sct 解析相关推荐

  1. KEIl工具之scatter file分散加载文件1

    KEIl工具之scatter file分散加载文件 转自:http://blog.csdn.net/zhoujiaxq/article/details/8102587 **************** ...

  2. KEIL MDK链接脚本-分散加载文件sct

    在了解keil的链接脚本之前需要了解几个重要概念: RO(ReadOnly):表示程序中的指令和常量 RW(Read/Write):表示程序中已初始化的变量 ZI(Zero):表示程序中未初始化的变量 ...

  3. 【C语言常识】Keil MDK的分散加载文件.sct

    https://blog.csdn.net/wuhenyouyuyouyu/article/details/71171546?ops_request_misc=%257B%2522request%25 ...

  4. keil的sct文件_Keil sct分散加载文件

    博主是个还没入门的弱菜,老师让查资料所以我把自己找的资料整理一下搁在这里方便以后查阅用的,自己并没有试过. 如有错误,欢迎指正. 参考资料: 首先介绍几个概念: 1.ARM映像文件 ARM映像文件是一 ...

  5. keil分散加载文件sct写法

    //-------------------------------------------------------------------------------------------------- ...

  6. 【IoT】STM32 分散加载文件 .sct 解析

    1.STM32 启动文件与 .sct 文件分析 1) 定义STACK段,{NOINIT,读写}:分配一段内存大小为0.5K; 2) 定义HEAP段, {NOINIT,读写}:分配一段内存大小为1K; ...

  7. H750移植rt_thread操作系统完整工程分享,包括外部FLASH分散加载文件

    一.移植注意事项 1.在运行外部FLASH存储的代码之前首先要初始化QSPI进入内存映射模式,参考代码: //QSPI进入内存映射模式(执行QSPI代码必备前提,为了减少引入的文件, //除了GPIO ...

  8. app与bootloader共享内存的方法(分散加载文件)

    app要升级时要通知bootloader,然后进入bootloader模式,app要如何通知bootloader呢?以前用了写入升级标志到eeprom的方式,然后bootloader再去读取,这是一种 ...

  9. KEIL的分散加载文件

    KEIL的分散加载文件 使用分散文件指定栈和堆 创建root执行区 使用 FIXED 属性创建根区域 在特定地址放置函数和数据 使用分散加载显式放置命名部分 使用.ANY模块选择器放置未分配的段 使用 ...

最新文章

  1. RDKit | 基于RDKit描述三维分子形状(3D描述符)
  2. git grade 版本下载及安装
  3. Python3.5.2官方文档学习备忘录
  4. JS 在火狐浏览器下关闭弹窗
  5. 二次元日系游戏制作工具 - live2dSDK入门教程
  6. OAuth认证实现机制及单点登录原理
  7. 国企“造船”转行测试,成功拿下11K,如今谁又甘心平庸呢?
  8. vscode 如何快速跳出括号
  9. monkey命令——压力测试——转载参考01
  10. 沧小海读《图解TCP/IP》笔记——第一章 网络基础知识
  11. 基于思维导图的研究生创新能力培养
  12. 极客日报第118期:京东被曝显卡售后不肯维修要原价退款;​IBM发布第一个2纳米芯片;Bootstrap 5.0.0发布
  13. 数据挖掘与数据分析项目链家租房数据(三)进一步探索与归纳
  14. Android--微信支付
  15. 2013年10月17日浙大ZJG听百度CEO李彦宏讲座
  16. 计算机体系结构 第七章 网络
  17. 突破隔离合同线上签—电子签名的用武之地
  18. windows2007 iis安装
  19. c语言程序设计a考试题,C语言程序设计模拟试题A(附答案).doc
  20. java计算限流工具

热门文章

  1. MacOS 磁盘管理工具 diskutil 介绍
  2. 关于背单词的一点个人体会 (好文章)
  3. EasyNVR depends on ffmpeg,yasm/nasm not found or too old. Use --disable-yasm for a crippledbuild
  4. 苹果手机打不开html,苹果手机打不开app是怎么回事(苹果app异常解决方法)
  5. 【小程序源码】字体设计符号组合多功能
  6. 《关爱码农成长计划》第一期报告
  7. web自动化_selenium IDE安装与运行总结
  8. 多线程:模仿火车站售票
  9. 计算机网络:网络层——网际协议IP
  10. VBA实现KMP和LCS算法