Chinaunix首页 | 论坛 | 博客
  • 博客访问: 1262395
  • 博文数量: 548
  • 博客积分: 7597
  • 博客等级: 少将
  • 技术积分: 4224
  • 用 户 组: 普通用户
  • 注册时间: 2010-12-15 13:21
个人简介

嵌入式软件工程师&&太极拳

文章分类

全部博文(548)

文章存档

2014年(10)

2013年(76)

2012年(175)

2011年(287)

分类: LINUX

2011-02-20 19:35:47

ATPCS

===================

ATPCS是: Arm Thumb procedure Call Standard的缩写。意思是arm thumb子程序调用规范。

C语言函数与C函数之间进行调用是用同一个C函数调用方式进行的。如果我们要在汇编指令行中调用C函数,那么我们就要知道我们如何给C函数传递参数,哪个寄存器表示什么意思,C中的数据栈是如何使用的,我们如何获得函数的返回值。同样,如果我们用汇编代码写的汇编函数,要让C语言能调用的话,我们的汇编函数也需要按照C语言的函数编译成汇编的形式进行书写。因此,ATPCS就是这样一种规范子程序相互调用的规则。


寄存器的使用规则:

	寄存器使用规则,
				
		在ARM处理器中的每种模式下有16个基本的寄存器,从r0-r15。ATPCS规定了个个寄存器的用途和命名。

		r15, 也叫 pc, 程序计数器。 始终指向下一条即将要执行的指令的地址。(相当于X86平台下面的ip寄存器。)
		r14, 也叫lr, 连接返回寄存器。当我们调用一个函数的时候,我们需要保存当前函数执行的地址,lr就是用于保存当前执行的地址的寄存器,用于执行完调用函数之后返回用。
		r13, 也叫sp, 程序栈指针寄存器。
		r12, 也叫ip, 用于临时存储
		r11, 也叫fp, 程序帧指针,

		r10到r0用于局部变量和参数传递用。	但是,如果在函数中改动了r4到r14的寄存器的值的话,需要进行先保存,后使用,最后函数退出前恢复。

	数据栈使用规则,

		ARM有4种数据栈,分别是:

			递增空栈,
			递增满栈,
			递减空栈,	
			递减满栈,

		ATPCS规定数据栈为递减满栈。

	参数传递规则,

		当参数个数小于等于4个的时候,使用r0到r3这4个寄存器进行参数传递;如果参数个数大于4个,余下的参数就通过sp所指向的数据栈进行参数传递。比如有3个参数的话,那么r0代表函数的第一个参数,r1代表函数的第二个参数,r2代表函数的第三个参数。比如有6个参数的话,那么r0-r3表示前面4个参数,然后余下的两个参数通过在栈上开辟8字节的空间进行参数传递。

	函数的返回值,
		
		函数调用完毕后,如果函数有返回值,函数一般把返回值保存在r0寄存器中,因此一般我们通过bl指令调用一个函数后,就可以通过在汇编里面访问r0得到返回值。


举个例子:

A:通过汇编语言调用C语言的函数:

	比如我要通过汇编语言调用下面的add函数,下面是add函数的代码,

	int add( int a, int b , int c )
	{
		return a + b + c;
	}	

	比如我要用汇编语言作等同于add( 3, 4, 5 );的函数调用,汇编代码如下,

	mov r0, #3
	mov r1, #4
	mov r2, #5
	bl add

B:通过C语言调用用汇编些的函数:

	下面是通过汇编实现的函数,可以通过C进行调用。

	.global add		# 申明此函数符号为全局
add:					# 函数标签名
	add r1, r1, r2	# 把C传过来的第二个参数加上第三个参数结果存入r1中。
	add r0, r0, r1	# 把第一个参数于上面的r1相加结果存在r0中,同时用做返回值。
	mov pc, lr		# 返回调用函数

	我们可以把上面代码保存为一个add.s, 然后通过arm-linux-gcc -c add.s -o add.o 生成目标代码。然后在如下的C程序中调用,


	int main( )
	{
		int result = add( 12, 34, 34 );
		printf("result = %d\n", result );
		return 0;
	}
		
	假如此C程序命名为main.c,那么联合上面的add.o编译为: gcc main.c add.o -o main。

	执行./main, 输出80


阅读(554) | 评论(0) | 转发(0) |
0

上一篇:ARM指令

下一篇:ARM内嵌汇编

给主人留下些什么吧!~~