当前位置:首页 > 嵌入式 > 嵌入式硬件
[导读]上一节介绍了如何使用Scatter文件对程序的代码和数据进行放置。但这些方法只有在外设和堆栈限制在源文件或头文件中定义好的前提下才能使用。为了增加程序的灵活性,最好在Scatter文件中设置这些信息,本节将介绍这些方法。

13.6 进一步存储器映射考虑事项

上一节介绍了如何使用Scatter文件对程序的代码和数据进行放置。但这些方法只有在外设和堆栈限制在源文件或头文件中定义好的前提下才能使用。为了增加程序的灵活性,最好在Scatter文件中设置这些信息,本节将介绍这些方法。

13.6.1 在Scatter文件中定位目标外设

通常情况下,外设寄存器的内存映射地址是在源文件或头文件中定义的“硬编码(hard-code)”。但为了增加代码的可移植性,可以在源文件中声明一个映射到外设寄存器的结构,并在这个结构在Scatter文件中定位。

下面的例子定义了映射32位寄存器的定时器的C结构。

struct {

volatile unsigned ctrl; /* timer 控制寄存器 */

volatile unsigned tmr; /* timer值 */

} timer_regs;

要把该结构放在存储器映射的特定地址,需创建一个新的执行区来装入该结构。

下面的例子说明了在Scatter文件中将timer_regs结构定位在0x40000000地址处。

ROM_LOAD 0x24000000 0x04000000

{

; ...

TIMER 0x40000000 UNINIT

{

timer_regs.o (+ZI)

}

; ...

}

需要特别注意的是,在应用程序启动过程中不将这些寄存器的内容初始化为零,因为这些地址对应的是外设寄存器,如果将其初始化为0,很可能改变系统的状态。必须将执行域的属性标记为UNINIT,这样可避免该区中的ZI数据初始化为零。

13.6.2 在Scatter文件中放置堆和栈

ARM公司建议,在Scatter文件中指定堆和栈的位置。这主要有两个主要优点:

· 有关存储器映射的所有信息保存在一个文件中;

· 改变堆和栈只要求重新链接,而不需要重新编译。

1.显示放置标号

为了在Scatter文件中放置堆栈,必须在源文件中定义Scatter文件的参照符号。下面的例子在名为stackheap.s的汇编文件中创建标有stack_base和heap_base的符号。这样就可以在Scatter文件的执行域中定位每个符号。

AREA stacks, DATA, NOINIT

EXPORT stack_base

stack_base SPACE 1

AREA heap, DATA, NOINIT

EXPORT heap_base

heap_base SPACE 1

END

下面的Scatter文件说明了如何在地址0x20000放置堆基址,如何在地址0x40000放置栈基址。堆基址和栈基址的位置可通过分别编辑其执行区予以改变。但该方法的缺点是在该栈区的上部占用一个字的内存区域放置SPACE(stack_base)变量。

LOAD_FLASH 0x24000000 0x04000000

{

; ...

HEAP 0x20000 UNINIT

{

stackheap.o (heap)

}

STACKS 0x40000 UNINIT

{

stackheap.o (stacks)

}

; ...

}

图13.17显示了堆和栈在内存中的放置情况。

图13.17 显示设置符号放置堆栈

2.使用链接程序生成符号

该方法需要在目标文件中指定堆和栈的大小。首先,在一个汇编源文件中为堆和栈定义一个适当大小的区域。使用SPACE命令保留一个清零的存储器块。然后,为该区域设置NOINIT属性,避免在链接时被修改。这样避免了显示放置堆栈标号而浪费内存空间。

下面的例子显示了如何在汇编源文件中预留出堆栈区域。

AREA stack, DATA, NOINIT

SPACE 0x3000 ;为栈预留的空间

AREA heap, DATA, NOINIT

SPACE 0x3000 ;为堆预留的空间

END

最后,可以在Scatter文件中定义执行域放置系统堆栈。

下面的例子显示了如何在Scatter文件中使用由联接器生成的符号放置堆栈。

LOAD_FLASH 0x24000000 0x04000000

{

:

STACK 0x1000 UNINIT ;length = 0x3000

{

stackheap.o (stack) ;stack = 0x4000 to 0x1000

}

HEAP 0x15000 UNINIT ;length = 0x3000

{

stackheap.o (heap) ;heap = 0x15000 to 0x18000

}

}

链接程序生成了指向每个执行区基址和限制的符号,可将其引入目标代码,供__user_initial_stackheap()函数使用:

Image$$STACK$$ZI$$Limit = 0x4000

Image$$STACK$$ZI$$Base = 0x1000

Image$$HEAP$$ZI$$Base = 0x15000

Image$$HEAP$$ZI$$Limit = 0x18000

下面的例子通过使用DCD伪操作赋予这些链接符号更有意义的名称,可使该代码可读性更高。

IMPORT ||Image$$STACKS$$ZI$$Base||

IMPORT ||Image$$STACKS$$ZI$$Limit||

IMPORT ||Image$$HEAP$$ZI$$Base||

IMPORT ||Image$$HEAP$$ZI$$Limit||

stack_base DCD ||Image$$STACKS$$ZI$$Limit|| ; = 0x4000

stack_limit DCD ||Image$$STACKS$$ZI$$Base|| ; = 0x1000

heap_base DCD ||Image$$HEAP$$ZI$$Base|| ; = 0x15000

heap_limit DCD ||Image$$HEAP$$ZI$$Limit|| ; = 0x18000

这样如果需要改变系统堆栈的设置,可以通过编辑Scatter文件中的执行域很容易地改变,而不需要重新编译源文件。

3.使用Scatter文件的EMPTY属性

该方法使用了Scatter文件执行域的EMPTY属性。该属性使得定义的区域不包括目标代码或数据。这是定义堆和栈的一个方便方法。区域的长度在EMPTY属性后指定。对于存储器中向上增长的堆,其区域的长度为正。对于栈,其长度被标为负数,说明其在存储器中是向下增长的。

下面的例子显示了如何在Scatter文件中使用EMPTY属性定义堆栈。

ROM_LOAD 0x24000000 0x04000000

{

...

HEAP 0x30000 EMPTY 0x3000

{

}

STACKS 0x40000 EMPTY -0x3000

{

}

...

}

该方法的优点是堆和栈的大小和位置是在一个地方定义的,即在Scatter文件中,而不必为堆栈创建stackheap.s源文件。

链接时,链接程序生成代表这些EMPTY区的符号。

Image$$HEAP$$ZI$$Base = 0x30000

Image$$HEAP$$ZI$$Limit = 0x33000

Image$$STACKS$$ZI$$Base = 0x3D000

Image$$STACKS$$ZI$$Limit = 0x40000

应用程序代码可处理这些符号,如下例所示。

IMPORT ||Image$$HEAP$$ZI$$Base||

IMPORT ||Image$$HEAP$$ZI$$Limit||

heap_base DCD ||Image$$HEAP$$ZI$$Base||

heap_limit DCD ||Image$$HEAP$$ZI$$Limit||

IMPORT ||Image$$STACKS$$ZI$$Base||

IMPORT ||Image$$STACKS$$ZI$$Limit||

stack_base DCD ||Image$$STACKS$$ZI$$Limit||

stack_limit DCD

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

在工业控制、汽车电子等实时性要求严苛的领域,中断风暴如同悬在系统头顶的达摩克利斯之剑——当多个高优先级中断密集触发时,传统MCU常因处理能力不足陷入瘫痪。ARM Cortex-M内核通过NVIC(嵌套向量中断控制器)的优...

关键字: 中断风暴 ARM

在现代电子设备的研发、生产与维护链条中,JTAG(Joint Test Action Group,联合测试行动组)技术如同一条"隐形桥梁",连接着开发者与芯片内部的核心逻辑。从最初解决电路板测试难题的工业标准,到如今成为...

关键字: JTAG ARM

当工程师将代码从经典的8051架构迁移至现代ARM Cortex-M系列时,常常会遇到因内存对齐规则差异导致的硬件异常或性能下降问题。以某物联网设备厂商的迁移案例为例,其将基于8051的温湿度传感器通信协议移植至STM3...

关键字: ARM 8051

在嵌入式系统开发中,传统软件断点依赖指令替换,易受优化代码或ROM存储限制,而JTAG调试器通过硬件断点与内存监控功能,可突破这些瓶颈,实现精准调试。本文结合ARM Cortex-M与RISC-V架构实践,解析JTAG在...

关键字: JTAG调试器 硬件断点设置 嵌入式软件

在资源受限的嵌入式系统中,软件崩溃往往导致设备失控、数据丢失甚至安全风险。本文梳理12种常见崩溃类型,结合典型场景与解决方案,帮助开发者构建更健壮的嵌入式系统。

关键字: 嵌入式软件 嵌入式系统

在嵌入式系统开发中,分层架构设计是平衡硬件依赖性与软件可维护性的核心方法。通过将系统划分为功能明确的层次,开发者可实现"关注点分离",使硬件变更不影响上层逻辑,软件迭代不干扰底层驱动。本文解析通用嵌入式架构的分层模型与实...

关键字: 嵌入式软件 嵌入式系统

在电子工程领域,JTAG(Joint Test Action Group)技术已成为芯片测试和系统调试的核心工具。从1980年代为解决PCB制造问题而诞生,到如今广泛应用于FPGA配置、嵌入式系统调试和芯片级编程,JTA...

关键字: JTAG ARM

嵌入式中间件与软总线作为现代分布式系统的核心基础设施,对于降低系统开发复杂度、实现异构环境互操作至关重要。文章系统梳理了应用服务器、远程过程调用(RPC)、消息中间件、容器编排平台等主流中间件以及新兴软总线技术的发展脉络...

关键字: 嵌入式软件 软件中间件 软总线 分布式系统

嵌入式系统作为针对各类特定用途的计算平台,已被广泛应用到航空航天、轨道交通、汽车电子以及智慧城市等领域。随着嵌入式系统复杂度增加,传统嵌入式软件语言及开发面临性能瓶颈、资源受限、安全要求严苛及开发效率低下等多重挑战。传统...

关键字: 嵌入式软件 软件开发语言 低代码 虚拟仿真
关闭