HC32L110(四) HC32L110的startup启动文件和ld连接脚本
目录
- HC32L110(一) HC32L110芯片介绍和Win10下的烧录
- HC32L110(二) HC32L110在Ubuntu下的烧录
- HC32L110(三) HC32L110的GCC工具链和VSCode开发环境
- HC32L110(四) HC32L110的startup启动文件和ld连接脚本
以下介绍项目中的startup和ld文件, 以及HC32L110的启动机制
仓库地址: https://github.com/IOsetting/hc32l110-template
如果转载, 请注明出处.
关于
因为是面向 GCC Arm Embedded 工具链的版本, 所以 startup 代码和 ld 连接描述脚本都依据 GCC Arm 工具链的格式.
Startup文件说明
startup_hc32l110.c 文件位于 Libraries/CMSIS
// 为下面的 uint32_t 等类型引入定义
#include <stdint.h>
// 将 ptr_func_t 定义为函数指针
typedef void (*ptr_func_t)();
// 下面这三个 __data 开头的变量是一组, 用于载入变量预先定义的值. 这些地址在连接阶段, 根据区域的实际情况被赋值
// __data_start 是载入的目标起始地址
extern uint32_t __data_start;
// __data_end 是载入的目标结束地址
extern uint32_t __data_end;
// 载入值的来源
extern uint32_t __data_load;
// __bss 开头的变量, 代表启动时需要清零的变量, __bss_start 和 __bss_end 分别代表了内存的起始和结束地址, 也是连接阶段会赋值
extern uint32_t __bss_start;
extern uint32_t __bss_end;
extern uint32_t __heap_start;
extern uint32_t __stacktop;
// 初始化, 在进入main函数之前需要执行的方法列表
extern ptr_func_t __init_array_start[];
extern ptr_func_t __init_array_end[];
// 引入外部定义的 SystemInit 和 main 方法
extern int main(void);
extern void SystemInit(void);
// 弱函数别名, 在对应的函数未定义时, 会调用别名对应的函数
#define WEAK_ALIAS(x) __attribute__ ((weak, alias(#x)))
// 下面这些都是中断函数
/* Cortex M3 core interrupt handlers */
void Reset_Handler(void);
void NMI_Handler(void) WEAK_ALIAS(Dummy_Handler);
void HardFault_Handler(void) WEAK_ALIAS(Dummy_Handler);
void SVC_Handler(void) WEAK_ALIAS(Dummy_Handler);
void PendSV_Handler(void) WEAK_ALIAS(Dummy_Handler);
void SysTick_Handler(void) WEAK_ALIAS(Dummy_Handler);
// 直接用中断号作为函数名, 具体的对应关系在 ddl.h 中,
// 这些是沿用官方DDL驱动代码, 将来会替换为直接调用实际的中断处理函数
void IRQ000_Handler(void) WEAK_ALIAS(Dummy_Handler);
void IRQ001_Handler(void) WEAK_ALIAS(Dummy_Handler);
void IRQ002_Handler(void) WEAK_ALIAS(Dummy_Handler);
// 中间略
void IRQ031_Handler(void) WEAK_ALIAS(Dummy_Handler);
/* 将 __stacktop 初始地址记录到 __stack_init.
关于 used 的定义: 即是未被使用, 编译后也需要保留
This attribute, attached to a function, means that code must be emitted
for the function even if it appears that the function is not referenced.
This is useful, for example, when the function is referenced only in
inline assembly.
*/
__attribute__((section(".stack"), used)) uint32_t *__stack_init = &__stacktop;
/* Stack top and vector handler table
中断向量表, 这些函数, 和前面的定义需要一致. 这些函数的实际逻辑在 ddl.h 和 ddl.c中定义.
*/
__attribute__ ((section(".vectors"), used)) void *vector_table[] = {
Reset_Handler,
NMI_Handler,
HardFault_Handler,
0,
0,
0,
0,
0,
0,
0,
SVC_Handler,
0,
0,
PendSV_Handler,
SysTick_Handler,
IRQ000_Handler,
IRQ001_Handler,
IRQ002_Handler,
IRQ003_Handler,
// 中间略
IRQ029_Handler,
IRQ030_Handler,
IRQ031_Handler};
/*
最重要的, 重启后的初始化方法, 由ld文件中的 ENTRY(Reset_Handler) 指定
*/
__attribute__((used)) void Reset_Handler(void)
{
uint32_t *src, *dst;
/* 从 Flash 到 RAM 复制变量值 */
src = &__data_load;
dst = &__data_start;
while (dst < &__data_end) *dst++ = *src++;
/* 清空 bss section */
dst = &__bss_start;
while (dst < &__bss_end) *dst++ = 0;
// 这里调用前面声明的 SystemInit
SystemInit();
// 调用初始化函数列表
for (const ptr_func_t *f = __init_array_start; f < __init_array_end; f++)
{
(*f)();
}
// 调用前面声明的main
main();
}
// 默认的中断处理方法
void Dummy_Handler(void)
{
while (1);
}
LD文件说明
以hc32l110x4.ld为例
/* MEMORY 内存块配置, 格式为
MEMORY
{
name [(attr)] : ORIGIN = origin, LENGTH = len
…
}
*/
MEMORY
{
FLASH (rx): ORIGIN = 0x00000000, LENGTH = 16K
RAM (rwx): ORIGIN = 0x20000000, LENGTH = 2K
}
// 运行一个程序时第一个被执行到的指令称为"入口点", 默认是start, 可以使用"ENTRY"连接脚本命令来设置入口点.参数是一个符号名
ENTRY(Reset_Handler)
/* "SECTIONS"命令是链接脚本中最重要的部分, 段命令格式如下, 会包含多个 secname, 区域必须已经在MEMORY中定义
SECTIONS {
...
secname start BLOCK(align) (NOLOAD) : AT ( ldadr )
{ contents } >region :phdr =fill
...
}
*/
SECTIONS
{
// 当前地址为FLASH区域起始地址
. = ORIGIN(FLASH);
//
.text : {
// KEEP 命令主要作用是防止垃圾收集机制把这几个重要的节排除在外,另外也保证堆栈和向量表在段中的位置处于最顶端
KEEP(*(.stack))
// 对应startup里面的 section(".vectors")
KEEP(*(.vectors))
KEEP(*(.vectors*))
// .text: 所有的编译出来的代码段,都放在这里
KEEP(*(.text))
// 通过 ALIGN 命令, 将当前地址指针调整到4字节对齐
. = ALIGN(4);
*(.text*)
. = ALIGN(4);
// 常量数据的代码段
KEEP(*(.rodata))
*(.rodata*)
// 当前指针, 调节到4字节对齐后的地址
. = ALIGN(4);
} >FLASH // 这个段放在名为FLASH的内存块
// 初始化方法指针队列
.init_array ALIGN(4): {
__init_array_start = .;
KEEP(*(.init_array))
__init_array_end = .;
} >FLASH
__stacktop = ORIGIN(RAM) + LENGTH(RAM);
// LOADADDR(.data) 获取.data段的加载地址(lma),也就是data段在Flash中存放的起始地址
__data_load = LOADADDR(.data);
// 当前地址为 RAM 区域起始地址
. = ORIGIN(RAM);
// 数据部分, 可以看下面对 __data_start 和 __data_end 的赋值方式
.data ALIGN(4) : {
__data_start = .;
*(.data)
*(.data*)
. = ALIGN(4);
__data_end = .;
} >RAM AT >FLASH // 这些变量位于RAM, 值会从FLASH的对应区域载入
/* 可以看下面对 __bss_start 和 __bss_end 的赋值方式
关于 NOLOAD: The `(NOLOAD)' directive will mark a section to not be loaded
at run time. The linker will process the section normally, but will mark
it so that a program loader will not load it into memory
就是会正常连接, 但是运行时不载入内存
*/
.bss ALIGN(4) (NOLOAD) : {
__bss_start = .;
*(.bss)
*(.bss*)
. = ALIGN(4);
__bss_end = .;
*(.noinit)
*(.noinit*)
} >RAM // 这些变量位于RAM
. = ALIGN(4);
__heap_start = .;
}
参考
- GNU linker ld (GNU Binutils) version 2.39 https://sourceware.org/binutils/docs-2.39/ld/index.html
- GCC链接脚本(.ld)文件详解 https://blog.csdn.net/a2529280665/article/details/121576020
HC32L110(四) HC32L110的startup启动文件和ld连接脚本的更多相关文章
- [ARM] Cortex-M Startup.s启动文件相关代码解释
1. 定义一个段名为CSTACK, 这里: NOROOT表示如何定义的段没有被关联,那么同意会被优化掉,如果不想被优化掉就使用ROOT. 后面的括号里数字表示如下: (1):这个段是2的1次方即2字节 ...
- STM32启动文件详细解析(V3.5.0) 以:startup_stm32f10x_hd.s为例
我用的是IAR,这个貌似是MDK的,不过很有用,大家可以看一下 ;* 文件名 : startup_stm32f10x_hd.s ;* 库版本 : V3.5.0 ;* 说明: 此文件为STM32F10x ...
- win10启动文件夹:
win10启动文件夹: C:\ProgramData\Microsoft\Windows\Start Menu\Programs\StartUp
- stm32启动文件 startup_stm32f10x_hd.s
;* 文件名 : startup_stm32f10x_hd.s;* 库版本 : V3.5.0;* 说明: 此文件为STM32F10x高密度 ...
- STM32启动过程--启动文件--分析
一.概述 1.说明 每一款芯片的启动文件都值得去研究,因为它可是你的程序跑的最初一段路,不可以不知道.通过了解启动文件,我们可以体会到处理器的架构.指令集.中断向量安排等内容,是非常值得玩味的. ST ...
- STM32启动文件的选择
移植了同事一个程序,然后死活不能用,发现启动文件错了,明天继续调.真把人折腾死了. stm32给的库文件太琐碎了,正如它的芯片型号一样繁多,例如启动文件: 网上查到的各个文件的解释是: startup ...
- Cortex-M3 .s启动文件分析
1. 基本概念(CMSIS): Cortex Micro-controller Software Interface Standard,微控制器软件接口标准. 2. CMSIS标准的文件结构: a) ...
- startup_LPC17XX.s 启动文件分析
工程中startup_LPC17XX.s是M3的启动文件,启动文件由汇编语言写的,它的作用一般是下面这几个: 1)堆和栈的初始化 2)中断向量表定义 3)地址重映射及中断向量表的转移 4)设置系统时钟 ...
- stm32启动文件ld md hd cl vl xl分析及选择
startup_stm32f10x_cl.s互联型的STM32F105xx,STM32F107xxstartup_stm32f10x_hd.s 大容量的STM32F101xx,STM32F102xx, ...
随机推荐
- WTM框架使用技巧之:CI/CD(持续集成/持续部署)
1. 什么是WTM框架? 一个快速.灵活.社区活跃.最最最最高效的.netcore 后台管理系统.详见 https://wtmdoc.walkingtec.cn/ 欢迎大家付费支持WTMPlus,反哺 ...
- 记安装AWVS14过程踩的坑
由于之前的AWVS14用着用着无法扫描了,一扫就是失败,一气之下就重装系统了.重装系统后发现安装还是不行,折腾了好久,终于找到方法了. 安装acunetix_14.1.210324124.exe 没啥 ...
- <%= %> <%- %> <% %>是什么意思?
.ejs文件后缀的数据渲染,这是服务器端的.把 .html改成 .ejs, (1)<%= %>相当于html中的innerTEXT,导出不包含标签 . (2)<%- %>相当于 ...
- 无语——真的好用到不行的7个Python小技巧
本文总结了我几个我在学习python过程中,用到的几个超好用的操作,这里分享给大家,我相信你们也会非常喜欢,目录如下.这里提前索要再看,记得点一点再看哦.这只是其中一些技巧,以后会慢慢和大家分享. 1 ...
- gitlab和jenkins做持续集成构建教程
背景介绍 上一个轮回,我花了三篇文章的时间着重向大家介绍了在条件有限的情况下,如果优雅地进行前端发版和迭代.庆七一,热烈庆祝香港回归,人民生活水平越来越好,昨天上午我自掏腰包买了台服务器,决定由冷兵器 ...
- SHT11和SHT21传感器
1.传感器概述 SHT11和SHT21为瑞士Sensirion公司生产,精度和测量范围较广,但价格较高.SHT11和SHT21是具有IIC总线接口的单片全校准数字式相对湿度和温度传感器.该传感器采用独 ...
- [ERROR] Another process with pid 914 is using unix socket file.
mysql启动报错 1.首先到mysql的配置文件中,确定socket文件路径 vim /etc/my.cnf 2.删除mysql.sock.lock 3.启动mysql
- 零基础学Java(7)大数
大数 如果基本的整数和浮点数精度不能够满足需求,那么可以使用java.math包中两个很有用的类:BigInteger和BigDecimal.这两个类可以处理包含任意长度数字序列的数值.BigInte ...
- 4-4 Spring Test
Spring Test Ⅰ.主要解决的问题 使用SpringTest前 手动加载Sping配置 手动从Spring容器中获取对象 使用SpringTest后 只需要通过注解指定Spring配置类 在S ...
- Github隐藏使用技巧(超详解)
目录 github使用说明 查看别人的主页和项目 上传自己的项目 使用git下载github上的文件 使用git实现代码管理 使用git恢复被修改的文件 更多关于git使用小技巧 github使用说明 ...