以前稍微写过操作系统上的C程序,感受不出来:BSS段,堆栈的意义。到了在单片机上写程序也没有考虑这些问题。但是到了ARM上环境似乎没有那么简单了,C的环境要自己来创建,不然就不能用。这也深刻的感受到了C语言中原来难以理解的概念。
裸机建立C语言环境-设置堆栈指针
这个是使用C语言的首要条件,不过这个就是指定一个sp指针就可以了,很简单的。ldr sp, =4096。
裸机建立C语言环境-清理BSS段
如果C语言中用到的全局变量或者静态变量,这个编译的时候是把它们放到了BSS段,这个段在内存中。怎么建成的?手动写一个链接脚本,添加__bss_start __bss_end变量来表示BSS段的开始和结束。如下:
SECTIONS {
. = 0x00000000;
.text : { *(.text) }
.rodata ALIGN(4) : AT((LOADADDR(.text)+SIZEOF(.text)+3)&~(0x03)) {*(.rodata*)}
.data ALIGN(4) : AT((LOADADDR(.rodata)+SIZEOF(.rodata)+3)&~(0x03)) { *(.data) }
__bss_start = .;
.bss ALIGN(4) : { *(.bss) *(COMMON) }
__bss_end = .;
}
这样在应用程序中清理__bss_start到__bss_end之间内在中的内容。这样全局变量就可以用了,否则会出现异常。我遇到的具体表现为:全局变量的值无法更改。代码可以学习u-boot中汇编方法清理:
/*
* These are defined in the board-specific linker script.
*/
.globl _bss_start
_bss_start:
.word __bss_start
.globl _bss_end
_bss_end:
.word __bss_end
/*
* 清BSS段
*/
clear_bss:
ldr r0, _bss_start /* find start of bss segment */
ldr r1, _bss_end /* stop here */
mov r2, #0x00000000 /* clear */
clbss_l:str r2, [r0] /* clear loop... */
add r0, r0, #4
cmp r0, r1
ble clbss_l
mov pc, lr
/* end_of clear_bss */
也可以用C语言来实现:
void clean_bss(void)
{
extern int __bss_start, __bss_end;
int *p = &__bss_start;
for (; p < &__bss_end; p++)
*p = 0;
}
总结:就是往这段内存中写0.
设计资源 培训 开发板 精华推荐
- 分立元件搭建555定时器
- LTC3265IFE 低噪声 +7V/-2V 电源的典型应用电路来自一个单端 5V 输入电源(频率 = 200kHz)
- 用于便携式的 12V DC 到 DC 单输出电源
- 用于固定电压的 NCV4276C 400 mA 低压降稳压器的典型应用
- 使用PCB制做8段数码管
- 用于确定 L6566BTR 的 OVP 和 OCP 的 IC 行为的典型外部电路 SMPS 电源的多模式控制器
- AD587JRZ 作为 CMOS 双路 DAC 的 10V 基准的典型应用电路
- LT8330ES6 4V 至 36V 输入、-12V 反相转换器的典型应用电路
- LV8702VGEVB,LV8702V 2 通道全桥驱动器评估板
- BP5220A 具有输出开/关控制的降压DC/DC转换器的典型应用电路