朱老师ARM裸机学习笔记(七):汇编写启动代码之调用C语言
扫描二维码
随时随地手机看文章
实验环境说明
硬件平台:飞凌嵌入式OK210开发板(S5PV210)
说明:开发板已经刷入OK210原厂u-boot,本文程序直接下载进CPU iRAm中运行
软件平台:Ubuntu15.10 + arm-linux-gcc 4.4.3
1、C语言运行需要栈空间来存储临时变量和函数调用时的所占用的空间。我们在操作系统下写的应用程序以及其他一些单片机程序不需要设置栈是因为,操作系统或者单片机内置的代码已经初始化过栈空间,所以我们可以直接使用,但是ARM需要我们自己初始化栈。
2、栈又分为,满减栈,满增栈,空增栈和空减栈之分。而ATPCS(ARM-THUMB procedure call standard(ARM-Thumb过程调用标准)的简称,PCS规定了应用程序的函数可以如何分开地写,分开地编译,最后将它们连接在一起,所以它实际上定义了一套有关过程(函数)调用者与被调用者之间的协议。)规定ARM使用的是满减栈。所以初始化栈就是把栈指针指向栈的最高位地址即可。
3、S5PV210的CPU共有6个sp(r14)寄存器,即每种处理器模式下一个sp指针,也就是说每种处理器模式下都有一个栈。但是处理器复位后默认进入SVC模式,因此我们只用初始化SVC下的sp指针即可。
4、由于裸机程序并没有初始化SDRAM所以栈只能设置为iRAm的内存空间,关于栈的分配《S5PV210_iROM_ApplicationNote_Preliminary_20091126》中已经给出建议,
所以初始化栈的代码如下
/*step1:setSVCstack*/ldrsp,=0xd0037d80123123调用C函数
设置好栈后可以通过bl指令调用C语言函数,例如
blled_blink11完整实验程序如下
Makefile
arm-linux-ld-Ttext0x0-oled.elf$^arm-linux-objcopy-Obinaryled.elfled.binarm-linux-objdump-Dled.elf>led_elf.dis%.o:%.Sarm-linux-gcc$(CFLAGS)-o$@$<-c%.o:%.carm-linux-gcc$(CFLAGS)-o$@$<-cclean:rm-rf*.o*.elf*.bin*.dis12345678910111213141234567891011121314
start.S
/**S5PV210汇编调用C语言*/.global_start_start:/*step1:setSVCstack*/ldrsp,=0xd0037d80/*step2:callcfunction*/blled_blink/*step3:while(1)*/b.1234567891011121312345678910111213
led.c
#defineGPH2_CON0xE0200C40#defineGPH2_DAT0xE0200C44#definerGPH2_CON*((volatileunsignedint*)GPH2_CON)#definerGPH2_DAT*((volatileunsignedint*)GPH2_DAT)voiddelay(void);voidled_blink(void){/*step1:setGPH2_CONmodetoGPIOoutput*//*GPH2_CON[0-3]all0b0001*/rGPH2_CON=0x01111;/*step2:loadoffintoGPH0_DAT*/rGPH2_DAT=0xFF;//alloffwhile(1){rGPH2_DAT=0X0;delay();rGPH2_DAT=0XFF;delay();}}voiddelay(void){volatileunsignedinti=60000;while(i--);}1234567891011121314151617181920212223242526272829303132333412345678910111213141516171819202122232425262728293031323334