历史上的今天
返回首页

历史上的今天

今天是:2025年02月07日(星期五)

正在发生

2021年02月07日 | STM32 启动代码分析详解

2021-02-07 来源:eefocus

1、堆栈存储器


堆栈存储区是在片上存储器中的SRAM(或RAM)中由用户自行开辟的一片数据存储区域,并且堆栈区的大小可根据用户的需要任意指定(只要不超过SRAM或RAM的大小),而堆栈区的位置由编译器指定分配。

 

Cortex-M3/M4处理器的堆栈指针SP是“满递减,空递增”,呈现向下逆生长的特点。 


堆栈区数据的存储特点是“先进后出,后进先出”。


这种特点是由堆栈指针的移动方式决定的,先入栈的数据对应的指针值比较大,后入栈的数据对应的指针值比较小,而出栈时堆栈指针的值是递增的,所以指针值大的数据当然后出栈。


堆栈的作用:


局部变量的存储、函数调用时函数或子程序间数据的传递(形参的保存,其实函数调用一旦结束被调函数中定义的所有局部变量单元就会被释放)、函数调用时现场数据的保存、中断发生时现场数据的保存。


2、启动代码作用


C 程序的的执行是从主函数main()开始的。


可是微控制器上电后,是如何寻找main()函数的呢?


显然,我们不可能从硬件上来寻找到main()函数的入口地址。


实际上,main()函数的入口地址是编译器在编译过程中分配的。


并且,从微控制器上电到main()函数执行前,微控制器有个启动的过程,这个启动过程正是启动代码执行的过程,时间非常短暂。


启动代码的作用: 


1)初始化堆栈指针 SP == _initial_sp; 


2)初始PC指针 ==Reset_Handler(复位处理程序); 


3)初始化中断向量表; 


4)配置系统时钟; 


5)调用C库_main()函数初始化用户堆栈,从而最终调用main()主函数去到C世界。


3、编写启动代码


一般标准的CMSIS启动代码都是汇编语言的,用C来编写启动代码之前,我们需要先构思一下C能不能用来编写启动代码?


a)正确的在存储空间中摆放异常&中断向量表;


b)分散加载&C Library的初始化(调用 __main函数);


c)堆栈初始化。


其中 a)功能在标准的汇编启动代码中是通过声明一个数据段(就是汇编指令DCD xxx)来实现的,这个我们在C中可以用数组来做,关键是要摆放到指定的位置上,这点需要用分散加载来配合,所以 a)应该可以实现;


其中 b)最好实现,直接调用__main就好了;


c),虽然通过读标准的CMSIS汇编启动代码也可以知道堆栈是怎样初始化的。


以LPC54608为例,早期的芯片会有不同的处理方式,比如大家都很熟悉的STM32F103系列,是需要启动代码与分散加载配合完成的。


3.1、异常&中断向量表


之前我们说这个可以用数组来实现,普通数组肯定不行,因为这些向量的本质是中断服务函数的入口,也就是“函数指针”所以这个数组必须得是函数指针数组:


所以我们先声明一个函数指针类型:


typedef void ( *__vector )( void );


然后定义一个函数指针型数组取名为__vector_table:


__vector __vector_table[] = {};

    这个数组好做,但是最重要的是摆放问题,必须确保这个向量表被摆放到Flash的0x00000000地址上去(在LPC54608这颗芯片中默认的向量表地址),这个才是问题的关键。


C与汇编是无法解决这个问题的,管程序在存储空间内摆放的是编译器开发环境中的链接器,指导链接器工作的是分散加载的脚本文件,这里需要分散加载的介入。 


比如大家常用的开发MCU的各种开发环境(Keil、IAR、Eclipse……等)其中真正起到编译作用的就是编译内核,Keil的编译内核是ARMCC(有时写成CCARM)、IAR编译内核是ICCARM、Eclipse一般大多是GCC,编译内核实际上是分成4个功能模块:


预处理器(Preprocessor)、编译器(Compiler)、汇编器(Assembler)、链接器(Linker),预处理器负责把各种头文件与对应的C文件结合在一起,负责删除注释、展开宏定义等工作;


编译器负责把C语言转换为对应的汇编语言;


汇编器负责把汇编语言转换为可重定位的目标文件*.o,这个就已经是二进制文件了;


链接器负责把所有.o文件链接成一个整体并分配真实的物理地址(就是上文反反复复提到的分散加载)。


如何摆放异常&中断向量表呢?


这里以Keil MDK为例(IAR、GCC在这部分语法不同,不过原理相同),指导链接器如何工作有一个专门的描述脚本文件,在Keil中此文件以.sct/.scf为扩展名,在这里可以找到分散加载描述文件:


这里直接列出来我针对自己写的启动代码编写的分散加载文件:


load_rom 0x00000000 0x00080000

{

    vector_rom 0x00000000 0x400

    {

        *( vector_table, +first)        

    }

    

    execute_rom 0x400 FIXED 0x0007FC00

    {

        *( InRoot$$Sections )

        .any( +ro )

    }

    

    execute_data 0x20000000 0x00010000   

    {

        .any ( +rw +zi )

    }

    

    ARM_LIB_HEAP  +0 empty 0x400 {}

    ARM_LIB_STACK 0x20020000  empty -400 {}

}

其中用*( vector_table, +first)中的+first来确保项链表被放置到Flash地址的最前端,这里需要注意的是,分散加载文件中的语言是脚本语言,既不是C也并非汇编,这个文件无法编译更无法调试,如果写错了,只能通过经验来修改不能DEBUG。


我们要放置一个数组,那么我们需要声明一个数据段Symbol,然后用分散加载来指定这个数据段Symbol的放置方式也就是+first。所以我们要先让这个数组生成一个段:


const __vector __vector_table[] __attribute__( ( section ( "vector_table" ), used ) ) = {};


于是我们使用__attribute__来实现。


堆栈的指定我们在分散加载中完成,即:


ARM_LIB_HEAP  +0 empty 0x400 {}

ARM_LIB_STACK 0x20020000  empty -400 {}

这两句话的意思就是声明一个大小为0x400的堆和一个大小为0x400的栈,注意堆和栈是不同的,简单来说函数的调用会占用栈空间,而使用malloc这样的函数则会产生堆分配,具体细节大家自己去百度。


*( InRoot$$Sections )


这部分指定了C Library以及分散加载部分代码的放置;


总之,这个分散加载文件最终把代码在存储空间内放置成了类似下图的形态:


 

__attribute__关键字用于指定函数、变量等Symbol的属性,是编译器可以识别的C语言关键字,当然这个要看使用的是什么编译器,不同的C编译器语法上略有差别,有的C编译器甚至还不支持__attribute__关键字,以下是Keil的帮助文档里面的关于__attribute__关键字可实现的功能:


__attribute__( ( section ( "vector_table" ), used ) ),其中的“section("vector_table")”意思是在elf(可重定位的目标文件)文件中放置一个名字为vector_table的section。used关键字指定了编译器该变量要在OBJ文件中保持为static类型。


再配合分散加载的*( vector_table, +first)的+first属性我们就成功的定义了一个放置在Flash最前端地址0x00000000地址上并且不会被编译器优化掉,数值不会被改掉,名字为vector_table的向量数据段。


这个是编译器识别的特殊栈标号,可以在分散加载或者汇编或者C语言中使用,0x00000000地址(向量表首地址)放置主堆栈指针MSP,0x00000004地址放置PC指针,所以我们先导入这个标号:


0x00000004放置PC指针,我们需要把复位向量放到此处:


3.2、__reset_handler复位函数


这个比较简单,最重要的是调用__main,有一些初始化要放到__main之前的,LPC54608的FPU以及一些SRAM的开关要放到__main之前;


首先导入__main函数:


extern void __main( void );


然后是__reset_handle的函数体:


void __reset_handler ( void )

{

#if ( ( __FPU_PRESENT == 1 ) && ( __FPU_USED == 1 ) )

    SCB->CPACR |= ( ( 3UL << 10 * 2 ) | ( 3UL << 11 * 2 ) );       // set CP10, CP11 Full Access 

#endif  

    SCB->VTOR = ( uint32_t ) 0x00000000;

    SYSCON->ARMTRACECLKDIV = 0;

    SYSCON->AHBCLKCTRLSET[ 0 ] = SYSCON_AHBCLKCTRL_SRAM1_MASK | SYSCON_AHBCLKCTRL_SRAM2_MASK | SYSCON_AHBCLKCTRL_SRAM3_MASK;

    

    __main();

}

因为是C语言的,在__main之前加入自己需要的函数体部分会更容易。


3.3、默认的异常&中断处理函数


以其中一个向量为例:(USB0中断)


函数体部分:


void usb0_irqhandler              ( void )                 

{

    while( 1 );

}   

然后再前面声明为“弱函数”,依然使用__attribute__来指定其属性为“weak”:


extern void usb0_irqhandler          ( void ) __attribute__( ( weak ) );    

    如果你不写中断服务函数,但程序却进了这个中断,程序就会进入默认的中断服务函数,如果你自己写了一个,你自己的中断服务函数会覆盖被声明为“weak”的函数,当然前提是函数名一致。



refer:


https://blog.csdn.net/cacti_one/article/details/72811281 

https://blog.csdn.net/weixin_39118482/article/details/79650305 


推荐阅读

史海拾趣

Goldentech Discrete Semiconductor Inc公司的发展小趣事

随着物联网、可穿戴设备等新兴领域的兴起,Goldentech敏锐地捕捉到了市场的新机遇。公司迅速调整战略方向,加大在微型化、低功耗半导体器件的研发投入。通过引入先进的制造工艺和封装技术,Goldentech成功推出了一系列适用于物联网和可穿戴设备的高性能离散半导体产品。这些产品凭借出色的性能和稳定性,在市场中获得了广泛认可,为公司的持续发展注入了新的动力。

DENWIRE公司的发展小趣事

为了进一步提升国际竞争力,Goldentech制定了明确的国际化战略。公司通过设立海外研发中心、销售网络和服务中心,加强与国际市场的联系和沟通。同时,Goldentech还积极参与国际标准的制定和推广工作,提升其在全球半导体行业的话语权和影响力。经过多年的努力,Goldentech已经成功在多个国家和地区建立了完善的业务体系和服务网络,为全球客户提供更加便捷、高效的服务和支持。

诚润电子(CHEVRON)公司的发展小趣事

诚润电子深知,一个优秀的团队是企业成功的关键。因此,他们一直致力于打造一支高素质、专业化的团队。公司注重员工的培训和发展,为员工提供广阔的职业发展空间和良好的福利待遇。同时,诚润电子还积极营造一种积极向上的企业文化氛围,让员工在工作中感受到归属感和成就感。这种良好的团队建设机制为诚润电子的持续发展提供了有力的保障。

Amulet Technologies公司的发展小趣事

在电子行业中,创新是企业持续发展的动力。诚润电子一直保持着对新技术、新产品的敏锐洞察力。他们不断投入研发资金,引进先进的生产设备和技术人才,致力于研发出更加先进、更加符合市场需求的产品。正是这种不断创新的精神,使得诚润电子在电子保护元件领域始终保持领先地位。

中移物联网(Chinamobile)公司的发展小趣事

2022年12月,中移物联网的OneNET城市物联网平台在2022中国移动全球合作伙伴大会上亮相。该平台以城市为核心,整合了各类物联网资源,为城市管理、公共服务等领域提供了全方位的物联网解决方案。OneNET城市物联网平台的推出,标志着中移物联网在物联网领域的技术实力和市场地位得到了进一步提升。

亿佰特(EBYTE)公司的发展小趣事

亿佰特(EBYTE)公司自2012年成立以来,一直致力于物联网通信技术的研发。公司团队凭借对无线通信技术的深入理解,不断突破技术瓶颈,成功研发出多款具有创新性的产品。这些产品不仅具备高性能和稳定性,而且能够广泛应用于智能家居、工业控制等领域。亿佰特通过持续的技术创新,逐步在电子行业中树立了领先地位。

问答坊 | AI 解惑

关于运放的虚短

请各位电子高手从内部结构上解释一下运放的“虚短”,谢谢!…

查看全部问答>

发布一块Renesas开发板的全部资料

Renesas单片机前身是三菱和日立单片机的联盟。因此每年出货量全球第一位。 设计了一块对此单片机的评估板,也可以说是一块开发板。(板子已经在制版,焊接好发图片,下周五之前) 因为不涉及公司机密,将在这个帖子里面发布我的所有资料。只是可惜 ...…

查看全部问答>

CE6.0 R3里的QQ和MSN支持语音聊天吗?

在有网络供使用的情况下,CE6.0 R3里的QQ和MSN支持语音聊天吗? 谢谢各位!…

查看全部问答>

DeviceIoControl与u盘硬件通信

利用DeviceIoControl()与u盘进行通信, 通信方式为IOCTL_SCSI_PASS_THROUGH_DIRECT, 这里需要一个CDB命令块,这个CDB命令是如何得到的。 我看到别人的程序中有BYTE Cdb[6] = {0x12,        0x01,      &n ...…

查看全部问答>

进制转化问题

数值的补码表示分两种情况: (1)正数的补码:与原码相同。 (2)负数的补码:符号位为1,其余位为该数绝对值的原码按位取反;然后整个数加1。 每当有人问我怎么把原码转化成补码时我就这样告诉他,自己也是这么求的。可是我不知道为什么要 ...…

查看全部问答>

请教TCPMP视频问题

请教各位DX:    我现在通过修改interface来实现自己设计的界面,由于修改了视频显示区域,所以原有视频显示不正常,有拉伸,如何修改原有视频数据,达到正常的缩放显示效果。…

查看全部问答>

为什么我不可以在MiniIDE里的output里做rd,t f之类debug?

我在miniIDE 里不能输入rd来注册输出,按不了回车。…

查看全部问答>

我的TIzigbee开发板不知为什么烧录不了呢?

我两块板上电了,设备管理器上也有显示,说明驱动也安装了。但是在debug时,弹出的选择目标(target selection)中是空白的。!!!! 那个对话框下面就写着 a target connot be selected wh ...…

查看全部问答>

2012电子大赛控制题讨论

本帖最后由 paulhyde 于 2014-9-15 04:05 编辑 现在清单都出来了,想必大家都研究了好几天了,大家过来讨论下,到底会出什么题呢?肯定不会太难,可是根据清单不得不往难处想,各种纠结啊,可以参考一下黑龙江激光打靶题,这题我也想了半天,除了 ...…

查看全部问答>